selftracker / app.py
Nakvi's picture
Upload 14 files
cd7bed1 verified
"""
FocusTrack - Streamlit Dashboard Entry Point
Run with: streamlit run app.py
"""
import sys
from pathlib import Path
sys.path.insert(0, str(Path(__file__).parent))
import streamlit as st
from ui_utils import DARK_CSS, LIGHT_CSS, get_db
# โ”€โ”€โ”€ Page Config (must be first Streamlit call) โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
st.set_page_config(
page_title="FocusTrack",
page_icon="๐ŸŽฏ",
layout="wide",
initial_sidebar_state="expanded",
)
# โ”€โ”€โ”€ Theme โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
db = get_db()
theme = db.get_setting("theme", "dark")
st.markdown(DARK_CSS if theme == "dark" else LIGHT_CSS, unsafe_allow_html=True)
# โ”€โ”€โ”€ Demo Mode Banner โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
import os
IS_HF = os.environ.get("SPACE_ID") is not None # True when running on HF Spaces
if IS_HF:
st.markdown("""
<div style="background:linear-gradient(90deg,#1a1a2e,#16213e);border:1px solid #6366f133;
border-radius:10px;padding:10px 16px;margin-bottom:1rem;font-size:0.82rem;
display:flex;align-items:center;gap:10px;color:#9090b0;">
<span style="font-size:1.1rem;">๐ŸŽฎ</span>
<span><strong style="color:#6366f1;">Demo Mode</strong> โ€” Live window tracking requires running locally on your laptop.
This demo shows 14 days of sample data. &nbsp;
<a href="https://github.com/yourname/focustrack" style="color:#22d3ee;">Get the full app โ†’</a></span>
</div>
""", unsafe_allow_html=True)
# โ”€โ”€โ”€ Sidebar โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
with st.sidebar:
# Logo
st.markdown("""
<div style="padding: 1rem 0 1.5rem; border-bottom: 1px solid #2a2a3a; margin-bottom: 1rem;">
<div style="display:flex; align-items:center; gap:10px;">
<div style="
width:36px; height:36px; border-radius:10px;
background: linear-gradient(135deg, #6366f1, #22d3ee);
display:flex; align-items:center; justify-content:center;
font-size:1.1rem; box-shadow: 0 4px 12px #6366f144;
">๐ŸŽฏ</div>
<div>
<div style="font-family:'Syne',sans-serif; font-weight:800; font-size:1.1rem; color:#f0f0ff; letter-spacing:-0.02em;">FocusTrack</div>
<div style="font-size:0.68rem; color:#5a5a7a; letter-spacing:0.08em; text-transform:uppercase;">Productivity Tracker</div>
</div>
</div>
</div>
""", unsafe_allow_html=True)
# Navigation
nav_items = {
"๐Ÿ  Live Dashboard": "dashboard",
"๐Ÿ“Š Analytics": "analytics",
"๐Ÿ“‹ Activity Log": "activity_log",
"โš™๏ธ Settings": "settings",
}
if "page" not in st.session_state:
st.session_state.page = "dashboard"
for label, key in nav_items.items():
is_active = st.session_state.page == key
if st.button(
label,
key=f"nav_{key}",
use_container_width=True,
):
st.session_state.page = key
st.rerun()
# Tracker status badge
st.markdown("<div style='margin-top:1rem; border-top:1px solid #2a2a3a; padding-top:1rem;'></div>",
unsafe_allow_html=True)
tracker_status = db.get_setting("tracker_running", "true")
status_map = {
"true": ("โ— Tracking", "#10b981"),
"paused": ("โ— Paused", "#f59e0b"),
"false": ("โ— Stopped", "#f43f5e"),
}
s_label, s_color = status_map.get(tracker_status, ("โ— Unknown", "#6366f1"))
st.markdown(
f"<div style='font-size:0.78rem; font-weight:600; color:{s_color}; "
f"padding:6px 12px; background:{s_color}11; border-radius:8px; "
f"border:1px solid {s_color}33; text-align:center;'>{s_label}</div>",
unsafe_allow_html=True,
)
# Privacy badge in sidebar
st.markdown("""
<div style="margin-top:auto; padding-top:2rem; font-size:0.7rem; color:#5a5a7a; text-align:center; line-height:1.5;">
๐Ÿ”’ 100% private<br>All data stays on your device
</div>
""", unsafe_allow_html=True)
# โ”€โ”€โ”€ Main Content โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
page = st.session_state.get("page", "dashboard")
if page == "dashboard":
from ui_pages.dashboard import render
render()
elif page == "analytics":
from ui_pages.analytics import render
render()
elif page == "activity_log":
from ui_pages.activity_log import render
render()
elif page == "settings":
from ui_pages.settings import render
render()