Skip to main content

A Streamlit custom component that creates a floating, draggable chat panel with glassmorphic design and automatic chat input management.

Reason this release was yanked:

docstrings

Project description

Streamlit Floating Container

A Streamlit custom component that creates a floating, draggable panel overlay for your Streamlit applications. Perfect for adding chat interfaces, FAQ sections, or any content that needs to be accessible without leaving the current page.

Features

  • Floating Panel — a glassmorphic floating panel that hovers over your Streamlit app
  • Draggable — vertical drag handle to reposition the panel button anywhere on the screen
  • Expandable — toggle the panel to nearly full viewport height
  • Start Position — configure the initial button position (top, middle, or bottom)
  • Auto-pinned Chat Inputst.chat_input placed directly inside the container automatically sticks to the bottom, mimicking Streamlit's native chat layout
  • Customizable — Material icon support, optional label, and glassmorphic toggle
  • Theme Integration — automatically adapts to your Streamlit theme colors
  • Resilient to reruns — component hot-updates in place from props; DOM state survives Streamlit reruns without reload hacks

Installation

pip install streamlit-floating-container

Or install from the local wheel:

pip install streamlit_floating_container-0.1.1-py3-none-any.whl

Usage

import streamlit as st
from streamlit_floating_container import FloatingContainer

st.set_page_config(layout="wide")
st.title("My App")

fp = FloatingContainer(
    icon=":material/chat:",
    label="Help",
    start_position="top",
    key="my_float",
    glassmorphic=True,
)

with fp.panel():
    st.subheader("FAQ", anchor=False)
    st.write("Your content here...")

Chat Example

Place st.chat_input directly inside the container — the component auto-pins it to the bottom of the panel (via the has-chat class on the scrollable wrapper) so it behaves like Streamlit's native chat layout.

import streamlit as st
from streamlit_floating_container import FloatingContainer

st.set_page_config(layout="wide")
st.title("My App")

if "chats" not in st.session_state:
    st.session_state["chats"] = []

def submit_chat(stkey):
    st.session_state["chats"].append(
        dict(who="user", message=st.session_state[stkey])
    )
    st.session_state["chats"].append(
        dict(who="ai", message="Response message")
    )

fp = FloatingContainer(
    icon=":material/chat:",
    label="Chat",
    start_position="middle",
    key="chat_float",
)

with fp.panel():
    # Message history
    for chat in st.session_state["chats"]:
        with st.chat_message(chat.get("who")):
            st.write(chat.get("message"))

    # Chat input placed directly in the container —
    # auto-pinned to the bottom of the panel.
    st.chat_input(
        "Type a message...",
        key="chat_input",
        on_submit=submit_chat,
        args=["chat_input"],
    )

Note: Only one st.chat_input widget is allowed per floating container. Additional chat inputs trigger a console warning and the panel will visually anchor the first one it finds.

FAQ / Static Content Example

import streamlit as st
from streamlit_floating_container import FloatingContainer

fp = FloatingContainer(
    icon=":material/help:",
    label="FAQ",
    start_position="bottom",
    key="faq",
    glassmorphic=False,  # solid background instead of frosted glass
)

with fp.panel():
    st.subheader("Frequently Asked Questions", anchor=False)
    st.markdown("**Q: How do I reset my password?**")
    st.markdown("A: Visit your profile page and click *Reset*.")
    st.divider()
    st.markdown("**Q: Who do I contact for support?**")
    st.markdown("A: Email support@example.com.")

API Reference

FloatingContainer

FloatingContainer(
    icon: str,
    label: str = "",
    start_position: Literal["top", "middle", "bottom"] = "top",
    key: str = "",
    glassmorphic: bool = True,
)

Parameters

Parameter Type Default Description
icon str required Material icon (e.g., :material/chat:) or single character
label str "" Label displayed in the panel header
start_position str "top" Initial button position: "top" (8%), "middle" (40%), or "bottom" (84%)
key str "" Unique key for the component instance
glassmorphic bool True Enable the frosted-glass / blur panel style

Only one FloatingContainer instance may be mounted per page. Attempting to mount a second one raises RuntimeError.

panel() context manager

with fp.panel():
    st.write("This appears in the floating panel")

panel() enters the panel's scrollable container directly — anything you render inside the with block appears in the floating panel. You do not need a second nested with statement.

If you render a st.chat_input(...) inside the panel, the component automatically pins it to the bottom. No manual placement required.

Panel Controls

Control Description
Open Button Click the floating button to open the panel
Close Button X button in the top-right corner
Expand / Collapse Toggle between normal size and near-fullscreen
Stretch Width Toggle the panel width between the default (400px) and stretched (800px)
Drag Handle Appears on hover on the floating button — drag vertically to reposition

Styling

The component uses CSS variables from your Streamlit theme:

  • --st-primary-color — button and accent colors
  • --st-secondary-background-color — drag handle background, pinned chat-input background
  • --st-text-color — text and icon colors

The panel features a glassmorphic design with blur effects and subtle shadows when glassmorphic=True. Set glassmorphic=False for a solid background that matches the theme.

Classes applied at runtime

Class Target Added when
panel-open #floating-panel, #close-panel Panel is open
expanded #floating-panel Panel is in expanded (near-fullscreen) mode
glassmorphic #floating-panel glassmorphic=True
has-chat #panel-scrollable A chat input is present in the container
scrollable-panel-is-empty #panel-scrollable No elements present
empty #input-div No elements in the fixed bottom slot
nav-open #open-panel Panel is open (used to hide the toggle button)
hidden #drag-handle Panel is open (hide drag handle)
active #movable-wrapper User is dragging the button

Implementation Notes

  • CCv2 component. Built on st.components.v2.component — the component is registered once at module import and mounted per render.
  • No reload handshake. The component hot-updates from data on every render; no full-page reload or localStorage prop diff.
  • DOM safety. Streamlit-managed nodes are never moved between trees; the component only toggles classes on them, avoiding React reconciler crashes (NotFoundError: insertBefore / removeChild) on reruns.
  • Observers. A MutationObserver watches the scrollable area to keep the has-chat class in sync with the actual presence of a chat input, across reruns and conditional renders.
  • Single-instance invariant. Mounting more than one FloatingContainer per page raises RuntimeError from Python.

File Structure

streamlit-floating-container/
├── pyproject.toml
├── README.md
└── streamlit_floating_container/
    ├── __init__.py
    ├── floating_container.html
    ├── floating_container.js
    └── styles.css

Development

git clone https://github.com/your-username/streamlit-floating-container.git
cd streamlit-floating-container
pip install -e .

Then run any demo script (e.g., main.py) with Streamlit:

streamlit run main.py

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

streamlit_floating_container-0.2.4.tar.gz (64.6 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

streamlit_floating_container-0.2.4-py3-none-any.whl (11.3 kB view details)

Uploaded Python 3

File details

Details for the file streamlit_floating_container-0.2.4.tar.gz.

File metadata

  • Download URL: streamlit_floating_container-0.2.4.tar.gz
  • Upload date:
  • Size: 64.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.27 {"installer":{"name":"uv","version":"0.9.27","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for streamlit_floating_container-0.2.4.tar.gz
Algorithm Hash digest
SHA256 dbee43bee76c2a37a8b62a0af5ba1251572a5a62beadf3698d5b8b6c929f9d7a
MD5 96f12f52358b18107aa7b237840a2416
BLAKE2b-256 ea2e493fdcac13947932f0da32fa26a2057f29f73794f3136ab162d382b55ecd

See more details on using hashes here.

File details

Details for the file streamlit_floating_container-0.2.4-py3-none-any.whl.

File metadata

  • Download URL: streamlit_floating_container-0.2.4-py3-none-any.whl
  • Upload date:
  • Size: 11.3 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.9.27 {"installer":{"name":"uv","version":"0.9.27","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"macOS","version":null,"id":null,"libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}

File hashes

Hashes for streamlit_floating_container-0.2.4-py3-none-any.whl
Algorithm Hash digest
SHA256 4f25e925c1862d727c05f98c1bc9473147c1dbdff711a9590182faef79067541
MD5 18b04ff1a07af3a391b0b23973b8673d
BLAKE2b-256 28280fa2e4054e8b34212eb0d75f4d4afdd6be772c8c8fa4315c655a1efba475

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page