"""Dark visual system: design tokens, the CDN `head` (fonts + Chart.js + flip/radar JS), and the global CSS. Single source of visual truth — backend integration never touches this. Why JS lives in `head` and not in `gr.HTML`: on Gradio 6, raw """ # JS handed to `.then(js=...)` after the result screen is shown, so the canvas exists & is sized. DRAW_RADAR_JS = "() => { setTimeout(function(){ window.__omcDrawRadar && window.__omcDrawRadar(); }, 70); }" CSS = f""" /* Center the app column on the page (horizontally) with comfortable top spacing. Gradio 6 nests content a few levels deep, so center every block-level wrapper inside the container, not just the container itself — otherwise the column can sit pinned to the left. */ .gradio-container {{ background: {BG} !important; max-width: 720px !important; margin: 0 auto !important; padding: 32px 20px 48px !important; }} .gradio-container .main, .gradio-container .wrap, .gradio-container .contain {{ margin-left: auto !important; margin-right: auto !important; }} .gradio-container, .gradio-container * {{ font-family: 'Inter', sans-serif; }} body {{ background: {BG}; }} footer {{ display: none !important; }} #screen-intake, #screen-processing, #screen-result {{ border: none; background: transparent; }} /* Headings / display type */ .omc-display {{ font-family: 'Saira', sans-serif !important; font-weight: 700; color: {TXT}; }} #omc-title {{ font-family: 'Saira', sans-serif; font-weight: 700; font-size: 30px; color: {TXT}; margin: 8px 0 6px; letter-spacing: .2px; }} #omc-subtitle {{ color: {TXT2}; font-size: 14px; line-height: 1.5; margin-bottom: 18px; }} /* Export-guide helper (provider buttons + revealed steps) */ .omc-guide-prompt {{ color: {TXT2}; font-size: 13px; margin: 10px 0 6px; }} #omc-prov-row {{ gap: 8px; }} .omc-prov-btn {{ font-family: 'Saira', sans-serif !important; font-weight: 600; background: {PANEL} !important; color: {TXT} !important; border: 1px solid {BORDER} !important; border-radius: 8px; }} .omc-prov-btn:hover {{ border-color: {TEAL} !important; color: {TEAL} !important; }} #omc-guide-panel .omc-guide {{ background: #12151C; border: 1px solid {BORDER}; border-radius: 10px; padding: 12px 14px; margin: 8px 0 4px; color: {TXT2}; font-size: 13px; line-height: 1.65; }} #omc-guide-panel .omc-guide h4 {{ font-family: 'Saira', sans-serif; color: {TXT}; font-size: 14px; margin: 0 0 6px; }} #omc-guide-panel .omc-guide ol {{ margin: 0; padding-left: 18px; }} #omc-guide-panel .omc-guide b {{ color: {TXT}; }} #omc-guide-panel .omc-guide code {{ background: {BG}; color: {TEAL}; padding: 1px 5px; border-radius: 4px; font-size: 12px; }} #omc-guide-panel .omc-guide .note {{ color: #E8B84B; opacity: .85; font-size: 11px; margin-top: 8px; }} /* Intake controls — match dark by color only, no heavy restyling */ #omc-file, #omc-name {{ background: {PANEL}; border: 1px solid {BORDER}; border-radius: 10px; }} #omc-file *, #omc-name * {{ color: {TXT}; }} #omc-file .wrap, #omc-file .file-preview {{ color: {TXT2}; }} .gradio-container input[type=text], .gradio-container textarea {{ background: {PANEL} !important; color: {TXT} !important; border-color: {BORDER} !important; }} #omc-analyze {{ background: {TEAL} !important; color: {BG} !important; font-family: 'Saira', sans-serif !important; font-weight: 600; border: none; border-radius: 10px; }} #omc-analyze:disabled, #omc-analyze[disabled] {{ background: #1d3a35 !important; color: {TXT3} !important; cursor: not-allowed; }} /* Processing screen */ #omc-proclog {{ min-height: 220px; }} .omc-proc-wrap {{ padding: 8px 2px; }} .omc-fact {{ display: flex; align-items: center; gap: 10px; color: {TXT}; font-size: 15px; padding: 9px 0; border-bottom: 1px solid {BORDER}; opacity: 0; animation: omcFadeIn .45s ease forwards; }} .omc-fact .dot {{ width: 7px; height: 7px; border-radius: 50%; background: {TEAL}; flex: none; }} .omc-fact .num {{ font-family: 'Saira', sans-serif; font-weight: 600; color: {TEAL}; }} .omc-fact.muted {{ color: {TXT2}; }} .omc-error {{ display: flex; align-items: flex-start; gap: 10px; color: #F8B4B4; font-size: 14px; line-height: 1.5; padding: 12px 14px; margin: 10px 0; background: rgba(220,80,80,.10); border: 1px solid rgba(220,80,80,.45); border-radius: 10px; }} .omc-error .dot {{ width: 7px; height: 7px; border-radius: 50%; background: #E25858; flex: none; margin-top: 6px; }} .omc-error b {{ color: #F3C0C0; font-family: 'Saira', sans-serif; }} .omc-langbar {{ height: 8px; border-radius: 4px; overflow: hidden; display: flex; width: 100%; margin-top: 6px; }} .omc-langbar .en {{ background: {TEAL}; }} .omc-langbar .other {{ background: {BORDER}; }} .omc-lang-legend {{ color: {TXT2}; font-size: 12px; margin-top: 6px; }} @keyframes omcFadeIn {{ from {{ opacity: 0; transform: translateY(4px); }} to {{ opacity: 1; transform: none; }} }} /* Live scoring progress bar (real, ticked per completed model call) */ .omc-progress {{ margin: 16px 0 4px; }} .omc-progress-track {{ height: 10px; background: #12151C; border: 1px solid {BORDER}; border-radius: 6px; overflow: hidden; }} .omc-progress-fill {{ height: 100%; background: linear-gradient(90deg, {TEAL}, #2BB89A); border-radius: 6px; transition: width .35s ease; box-shadow: 0 0 12px rgba(52,211,176,.45); }} .omc-progress-fill.indet {{ width: 35% !important; animation: omcIndet 1.1s ease-in-out infinite; }} .omc-progress-label {{ display: flex; justify-content: space-between; align-items: baseline; color: {TXT2}; font-size: 12px; margin-top: 7px; }} .omc-progress-label .num {{ font-family: 'Saira', sans-serif; font-weight: 600; color: {TEAL}; font-size: 14px; }} .omc-progress-label .pct {{ font-family: 'Saira', sans-serif; font-weight: 700; color: {TXT}; }} @keyframes omcIndet {{ 0% {{ margin-left: -35%; }} 100% {{ margin-left: 100%; }} }} /* Flip card — height increased from 480px to 560px so the radar (175px fixed) always fits alongside the avatar/name/tier/OVR/stars/notes even when the amber placeholder text and single-conversation scope note are both present. Both faces are position:absolute;inset:0 so they need a definite (not just min-) height on their parent to anchor bottom:0. */ .flip-card {{ perspective: 1200px; cursor: pointer; width: 100%; max-width: 380px; height: 560px; margin: 0 auto; }} .flip-card-inner {{ position: relative; width: 100%; height: 100%; transition: transform .6s; transform-style: preserve-3d; }} .flip-card.flipped .flip-card-inner {{ transform: rotateY(180deg); }} .flip-card-face {{ position: absolute; inset: 0; -webkit-backface-visibility: hidden; backface-visibility: hidden; background: {PANEL}; border: 1px solid {BORDER}; border-radius: 18px; padding: 22px; display: flex; flex-direction: column; box-shadow: 0 10px 40px rgba(0,0,0,.35); }} .flip-card-back {{ transform: rotateY(180deg); }} .flip-hint {{ position: absolute; bottom: 10px; right: 16px; color: {TXT3}; font-size: 11px; }} .omc-avatar {{ width: 64px; height: 64px; border-radius: 50%; background: {BG}; border: 1px solid {BORDER}; display: flex; align-items: center; justify-content: center; margin: 0 auto 10px; }} .omc-name {{ font-family: 'Saira', sans-serif; font-weight: 600; font-size: 20px; color: {TXT}; text-align: center; }} .omc-tier {{ font-family: 'Saira', sans-serif; font-weight: 700; font-size: 13px; text-align: center; display: inline-block; padding: 2px 12px; border-radius: 999px; margin: 8px auto 4px; }} .omc-tier-wrap {{ text-align: center; }} .omc-ovr {{ font-family: 'Saira', sans-serif; font-weight: 700; font-size: 46px; color: {TXT}; text-align: center; line-height: 1; margin: 6px 0 2px; }} .omc-ovr .slash {{ font-size: 18px; color: {TXT2}; font-weight: 600; }} .omc-stars {{ text-align: center; font-size: 22px; letter-spacing: 2px; margin: 4px 0 8px; }} .omc-star {{ position: relative; display: inline-block; color: {TXT3}; }} .omc-star .fill {{ position: absolute; left: 0; top: 0; overflow: hidden; color: #E8B84B; white-space: nowrap; }} /* Fixed height so Chart.js (maintainAspectRatio:false) always has a definite parent size. flex:1 was the bug: when placeholder + scope notes overflow the 480px card, flex:1 collapses to 0 and the canvas renders invisible / below the card. */ .omc-radar-wrap {{ height: 175px; min-height: 175px; position: relative; margin-top: 4px; flex: none; }} #omc-radar {{ width: 100% !important; height: 100% !important; display: block; }} /* Card back */ .omc-back-title {{ font-family: 'Saira', sans-serif; font-weight: 600; color: {TXT}; font-size: 15px; margin-bottom: 10px; }} .omc-bar-row {{ margin: 7px 0; }} .omc-bar-head {{ display: flex; justify-content: space-between; font-size: 12px; color: {TXT2}; margin-bottom: 3px; }} .omc-bar-head .v {{ font-family: 'Saira', sans-serif; color: {TXT}; }} .omc-bar {{ height: 7px; background: {BG}; border-radius: 4px; overflow: hidden; }} .omc-bar .fill {{ height: 100%; background: {TEAL}; border-radius: 4px; }} .omc-crit {{ font-size: 12px; color: {TXT2}; margin: 12px 0 4px; line-height: 1.6; }} .omc-crit b {{ color: {TXT}; font-family: 'Saira', sans-serif; font-weight: 600; }} .omc-conf {{ font-size: 11px; color: {TXT3}; margin-top: 6px; }} .omc-improve {{ font-size: 12px; color: {TEAL}; margin-top: auto; padding-top: 10px; line-height: 1.5; }} /* Honesty tag — amber when scores are a heuristic placeholder, teal when real ML */ .omc-placeholder {{ text-align: center; font-size: 10px; color: #E8B84B; opacity: .85; letter-spacing: .3px; margin: 4px 0; }} .omc-placeholder-sm {{ font-size: 10px; color: #E8B84B; opacity: .85; margin-bottom: 5px; }} .omc-real {{ text-align: center; font-size: 10px; color: {TEAL}; opacity: .85; letter-spacing: .3px; margin: 4px 0; }} .omc-real-sm {{ font-size: 10px; color: {TEAL}; opacity: .85; margin-bottom: 5px; }} /* Paste tab hint */ .omc-paste-hint {{ color: {TXT2}; font-size: 12px; line-height: 1.55; margin: 4px 0 8px; padding: 8px 10px; background: #12151C; border: 1px solid {BORDER}; border-radius: 8px; }} .omc-paste-hint b {{ color: {TXT}; }} #omc-paste-btn {{ background: {TEAL} !important; color: {BG} !important; font-family: 'Saira', sans-serif !important; font-weight: 600; border: none; border-radius: 10px; }} /* Single-conversation (paste) scope disclaimer */ .omc-scope {{ background: #12151C; border: 1px solid {BORDER}; border-radius: 8px; padding: 7px 10px; margin: 6px 0; text-align: center; }} .omc-scope b {{ display: block; font-family: 'Saira', sans-serif; color: #E8B84B; font-size: 11px; letter-spacing: .3px; margin-bottom: 2px; }} .omc-scope span {{ color: {TXT2}; font-size: 11px; line-height: 1.45; }} /* Evidence accordions */ .omc-evidence {{ color: {TXT2}; font-size: 13px; line-height: 1.6; }} .omc-evidence .q {{ display: block; color: {TXT}; border-left: 2px solid {TEAL}; padding: 2px 0 2px 10px; margin: 6px 0; }} .omc-evidence .tip {{ color: {TEAL}; margin-top: 8px; }} .omc-evidence .score {{ font-family: 'Saira', sans-serif; color: {TXT}; }} @media (prefers-reduced-motion: reduce) {{ .flip-card-inner {{ transition: none; }} .omc-fact {{ animation: none; opacity: 1; }} #omc-radar {{ }} }} """