# -*- coding: utf-8 -*- from __future__ import annotations import gradio as gr import plotly.graph_objects as go import numpy as np from datetime import datetime, timedelta import random import time import pandas as pd from typing import Callable, Any, List, Dict, Optional, Generator, Iterable, Tuple import re FOXAI_LOGO_URL = "" custom_css = """ .gradio-container { background: linear-gradient(135deg, #f8fafc 0%, #e2e8f0 25%, #cbd5e1 50%, #94a3b8 75%, #64748b 100%); font-family: 'Inter', 'Segoe UI', system-ui, -apple-system, sans-serif !important; min-height: 100vh; } .header-container { text-align: center; margin: 20px 0; padding: 40px; background: linear-gradient(90deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%); border-radius: 24px; box-shadow: 0 20px 40px rgba(12, 83, 135, 0.15), 0 8px 16px rgba(0, 0, 0, 0.1); backdrop-filter: blur(20px); border: 1px solid rgba(255, 255, 255, 0.3); position: relative; overflow: hidden; } .header-container::before { content: ''; position: absolute; top: 0; left: -100%; width: 100%; height: 100%; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.2), transparent); animation: shine 3s infinite; } @keyframes shine { 0% { left: -100%; } 100% { left: 100%; } } .foxai-logo { max-width: 320px; height: auto; margin: 15px auto 25px auto; display: block; filter: drop-shadow(0 4px 8px rgba(0,0,0,0.15)) brightness(1.1); transition: all 0.4s ease; animation: float 4s ease-in-out infinite; } .foxai-logo:hover { transform: scale(1.05); filter: drop-shadow(0 8px 16px rgba(12, 83, 135, 0.3)) brightness(1.2); } .header-title { background: linear-gradient(90deg, #ffffff, #f1f5f9, #e2e8f0, #ffffff); background-size: 300% 300%; animation: gradient 8s ease infinite; -webkit-background-clip: text; -webkit-text-fill-color: transparent; background-clip: text; text-align: center; font-size: 2.5em; font-weight: 700; margin: 15px auto; line-height: 1.3; text-shadow: 0 0 30px rgba(255, 255, 255, 0.5); letter-spacing: -0.01em; max-width: 900px; } @keyframes gradient { 0% { background-position: 0% 50%; } 50% { background-position: 100% 50%; } 100% { background-position: 0% 50%; } } /* Unified loading container styling */ .loading-container { text-align: center; padding: 40px; background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; border-radius: 20px; color: white !important; margin: 20px 0; box-shadow: 0 8px 25px rgba(12, 83, 135, 0.25) !important; border: 1px solid rgba(255, 255, 255, 0.3) !important; animation: loadingPulse 2s ease-in-out infinite alternate; } .loading-container * { color: white !important; } @keyframes loadingPulse { 0% { box-shadow: 0 15px 35px rgba(12, 83, 135, 0.3), 0 5px 15px rgba(0, 0, 0, 0.12); } 100% { box-shadow: 0 20px 45px rgba(12, 83, 135, 0.4), 0 8px 20px rgba(0, 0, 0, 0.15); } } .loading-spinner { border: 4px solid #f3f3f3; border-top: 4px solid #1A5F91; border-radius: 50%; width: 40px; height: 40px; animation: spin 1s linear infinite; margin: 0 auto 20px; } @keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } } .progress-bar { width: 100%; height: 20px; background-color: rgba(255,255,255,0.3); border-radius: 10px; overflow: hidden; margin: 15px 0; } .progress-fill { height: 100%; background: linear-gradient(90deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%); border-radius: 10px; animation: progress 3s ease-in-out; box-shadow: 0 0 20px rgba(12, 83, 135, 0.4); } @keyframes progress { 0% { width: 0%; } 25% { width: 30%; } 50% { width: 60%; } 75% { width: 85%; } 100% { width: 100%; } } .prediction-box { background: linear-gradient(90deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%); border-radius: 20px; padding: 25px; color: white !important; text-align: center; font-size: 1.1em; font-weight: 600; box-shadow: 0 15px 35px rgba(12, 83, 135, 0.25), 0 5px 15px rgba(0, 0, 0, 0.12); margin: 15px 0; border: 1px solid rgba(255,255,255,0.2); position: relative; overflow: hidden; } .prediction-box * { color: white !important; } .prediction-box::before { content: ''; position: absolute; top: 0; left: 0; right: 0; height: 1px; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.6), transparent); } .insight-box { background: linear-gradient(90deg, #cbd5e1 0%, #94a3b8 25%, #64748b 50%, #475569 75%, #334155 100%); border-radius: 20px; padding: 25px; color: white !important; margin: 15px 0; box-shadow: 0 15px 35px rgba(51, 65, 85, 0.25), 0 5px 15px rgba(0, 0, 0, 0.12); border: 1px solid rgba(255,255,255,0.2); position: relative; overflow: hidden; } .insight-box * { color: white !important; } .executive-summary { background: linear-gradient(90deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%); border-radius: 20px; padding: 30px; color: white !important; margin: 15px 0; box-shadow: 0 20px 40px rgba(12, 83, 135, 0.25), 0 8px 16px rgba(0, 0, 0, 0.12); border-left: 5px solid rgba(255, 255, 255, 0.3); border: 1px solid rgba(255,255,255,0.2); position: relative; overflow: hidden; } .executive-summary * { color: white !important; } .executive-summary::after { content: ''; position: absolute; top: 0; right: 0; width: 100px; height: 100%; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.1)); pointer-events: none; } .feature-box { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; color: white !important; border-radius: 20px !important; padding: 25px !important; margin: 15px 0 !important; box-shadow: 0 8px 32px rgba(12, 83, 135, 0.25), 0 4px 16px rgba(0, 0, 0, 0.1) !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; backdrop-filter: blur(10px) !important; transition: all 0.3s ease !important; text-align: center !important; } .feature-box:hover { transform: translateY(-2px) !important; box-shadow: 0 12px 40px rgba(12, 83, 135, 0.35), 0 6px 20px rgba(0, 0, 0, 0.15) !important; } .feature-box h3 { margin: 0 !important; color: white !important; font-weight: 700 !important; text-shadow: 0 0 10px rgba(255, 255, 255, 0.3) !important; font-size: 1.2em !important; text-align: center !important; } .feature-box * { color: white !important; } /* ALL components get unified gradient and white text */ .system-status, .metric-card { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; color: white !important; padding: 15px 25px; border-radius: 20px; margin: 15px 0; text-align: center; font-weight: 600; box-shadow: 0 8px 25px rgba(12, 83, 135, 0.25) !important; border: 1px solid rgba(255,255,255,0.3) !important; font-size: 0.95em; position: relative; overflow: hidden; } .system-status *, .metric-card * { color: white !important; } .metric-card:hover { transform: translateY(-3px); box-shadow: 0 12px 35px rgba(12, 83, 135, 0.3), 0 5px 12px rgba(0, 0, 0, 0.15); } .metric-card::before { content: ''; position: absolute; top: 0; left: 0; right: 0; height: 2px; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.8), transparent); } /* Unified button styling */ .button-predict, .btn-modern { background: linear-gradient(135deg, #1A5F91, #8B8B8B) !important; color: white !important; border: none !important; border-radius: 25px !important; padding: 15px 30px !important; font-size: 16px !important; font-weight: bold !important; box-shadow: 0 8px 25px rgba(26, 95, 145, 0.4) !important; transition: all 0.3s ease !important; position: relative; overflow: hidden; font-size: 0.95em; text-transform: uppercase; letter-spacing: 1.2px; } .button-predict:hover, .btn-modern:hover { transform: translateY(-2px) scale(1.02) !important; box-shadow: 0 12px 35px rgba(26, 95, 145, 0.6) !important; } .btn-modern::before { content: ''; position: absolute; top: 0; left: -100%; width: 100%; height: 100%; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.3), transparent); transition: left 0.6s; } .btn-modern:hover::before { left: 100%; } .plot-container { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; border-radius: 20px !important; padding: 15px !important; margin: 15px 0 !important; box-shadow: 0 8px 32px rgba(12, 83, 135, 0.25), 0 4px 16px rgba(0, 0, 0, 0.1) !important; height: 550px !important; max-height: 650px !important; min-height: 400px !important; position: relative !important; overflow: hidden !important; backdrop-filter: blur(10px) !important; } /* Simplify chart container - remove ALL extra nesting and borders */ .plot-container .plotly-graph-div { position: relative !important; width: 100% !important; height: 500px !important; max-height: 600px !important; margin: 0 !important; padding: 0 !important; border-radius: 8px !important; background: rgba(255, 255, 255, 0.98) !important; border: none !important; box-shadow: none !important; overflow: hidden !important; } /* Remove ALL additional containers and nested divs that cause multiple frames */ .plot-container .plotly-graph-div > div, .plot-container .plotly-graph-div > div > div, .plot-container .plotly-graph-div > div > div > div { background: transparent !important; border: none !important; box-shadow: none !important; padding: 0 !important; margin: 0 !important; outline: none !important; } /* Ẩn hoàn toàn toolbar/config của biểu đồ */ .plot-container .modebar, .plot-container .modebar-container, .plotly .modebar, .js-plotly-plot .modebar, .plotly-graph-div .modebar, .plotly-graph-div .modebar-container, .svg-container .modebar, div[data-title="Plotly toolbar"], .modebar-group, .modebar-btn { display: none !important; visibility: hidden !important; opacity: 0 !important; pointer-events: none !important; height: 0 !important; width: 0 !important; overflow: hidden !important; position: absolute !important; top: -9999px !important; left: -9999px !important; } /* Cải thiện spacing cho text và labels - tránh dính chữ */ .plot-container .plotly-graph-div .svg-container { pointer-events: auto !important; overflow: hidden !important; padding: 10px !important; height: 100% !important; max-height: 480px !important; } .plot-container .plotly-graph-div .main-svg { max-width: 100% !important; height: 100% !important; max-height: 500px !important; overflow: hidden !important; padding: 5px !important; } /* Đảm bảo text không bị cắt và có spacing tốt */ .plot-container .plotly-graph-div text { font-family: "Inter", "Segoe UI", Arial, sans-serif !important; font-size: 12px !important; text-anchor: middle !important; } .plot-container .plotly-graph-div .xtick text, .plot-container .plotly-graph-div .ytick text { font-size: 11px !important; fill: #374151 !important; } /* Cải thiện title spacing - tránh dính với nội dung */ .plot-container .plotly-graph-div .gtitle { font-size: 16px !important; font-weight: 600 !important; fill: #1f2937 !important; dominant-baseline: text-before-edge !important; } /* Tối ưu responsive và tránh overflow */ .plot-container { width: 100% !important; max-width: 100% !important; overflow: visible !important; } /* Đảm bảo axis labels có đủ space */ .plot-container .plotly-graph-div .xaxislayer-above, .plot-container .plotly-graph-div .yaxislayer-above { overflow: visible !important; } /* CSS cho smooth scrolling và highlight */ html { scroll-behavior: smooth; } .plot-container.highlight { box-shadow: 0 0 15px rgba(26, 95, 145, 0.2) !important; transform: scale(1.01) !important; transition: all 0.3s ease !important; } /* Hiệu ứng highlight cho loading panel */ #loading-panel.highlight { box-shadow: 0 0 20px rgba(26, 95, 145, 0.4) !important; transform: scale(1.02) !important; transition: all 0.3s ease !important; } /* Fade-in animation cho LLM analysis */ .gradio-markdown { animation: fadeInUp 0.8s ease-out !important; opacity: 0 !important; animation-fill-mode: forwards !important; } @keyframes fadeInUp { 0% { opacity: 0; transform: translateY(30px); } 100% { opacity: 1; transform: translateY(0); } } /* Smooth reveal cho các phần tử khác */ .plot-container, .info-card { animation: slideInFade 0.6s ease-out !important; } @keyframes slideInFade { 0% { opacity: 0; transform: translateY(20px); } 100% { opacity: 1; transform: translateY(0); } } /* Cải thiện focus state cho biểu đồ */ .plot-container:focus-within { outline: 2px solid rgba(26, 95, 145, 0.3); outline-offset: 2px; } /* Unified color scheme - standardize ALL backgrounds to the main gradient */ .info-card, .feature-box, .prediction-box, .insight-box, .executive-summary, .modern-card, .glass-card { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; color: white !important; border-radius: 20px !important; padding: 25px !important; margin: 15px 0 !important; border: 1px solid rgba(255, 255, 255, 0.3) !important; box-shadow: 0 8px 25px rgba(12, 83, 135, 0.25) !important; backdrop-filter: blur(10px) !important; } /* Ensure ALL text inside these containers is white */ .info-card *, .feature-box *, .prediction-box *, .insight-box *, .executive-summary *, .modern-card *, .glass-card * { color: white !important; } /* Unified header styling */ .info-card h3, .info-card h4, .feature-box h3, .prediction-box h3, .insight-box h3, .executive-summary h3, .modern-card h3, .glass-card h3 { color: white !important; text-shadow: 0 0 10px rgba(255, 255, 255, 0.3) !important; font-weight: 700 !important; } /* Unified text styling */ .info-card p, .info-card div, .feature-box p, .prediction-box p, .insight-box p, .executive-summary p, .modern-card p, .glass-card p { color: white !important; opacity: 0.95 !important; } /* Sửa lỗi font và text rendering */ * { font-family: 'Inter', 'Segoe UI', system-ui, -apple-system, sans-serif !important; text-rendering: optimizeLegibility !important; -webkit-font-smoothing: antialiased !important; -moz-osx-font-smoothing: grayscale !important; } /* Cải thiện hiển thị tiếng Việt và căn giữa */ .gradio-markdown, .gradio-html { font-family: 'Inter', 'Segoe UI', system-ui, -apple-system, sans-serif !important; line-height: 1.6 !important; word-wrap: break-word !important; overflow-wrap: break-word !important; text-align: center !important; } /* Căn giữa tất cả text */ .gradio-container .wrap { max-width: 100% !important; overflow-x: hidden !important; text-align: center !important; } /* Căn giữa các component */ .gradio-row { flex-wrap: wrap !important; justify-content: center !important; } .gradio-column { min-width: 0 !important; flex-shrink: 1 !important; display: flex !important; flex-direction: column !important; align-items: center !important; } /* Chuẩn hóa màu sắc - text đậm trên nền sáng */ .gradio-container { color: #1e293b !important; } /* Chuẩn hóa size chữ - màu đậm cho dễ đọc */ h1, h2, h3, h4, h5, h6 { font-weight: 600 !important; color: #1A5F91 !important; text-align: center !important; margin: 0.5em auto !important; } p, div, span { color: #334155 !important; font-size: 1em !important; line-height: 1.6 !important; font-weight: 500 !important; } /* Chuẩn hóa labels và inputs - màu đậm */ label { font-weight: 600 !important; color: #1A5F91 !important; text-align: center !important; margin-bottom: 8px !important; font-size: 1em !important; } .gradio-textbox, .gradio-dropdown, .gradio-radio { text-align: center !important; color: #1e293b !important; } /* Buttons styling */ .gradio-button { font-weight: 600 !important; font-size: 1em !important; border-radius: 12px !important; margin: 8px auto !important; display: block !important; color: #1e293b !important; } /* Styling cho markdown báo cáo chiến lược */ .gradio-markdown { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; color: white !important; border-radius: 20px !important; padding: 30px !important; margin: 20px 0 !important; box-shadow: 0 8px 32px rgba(12, 83, 135, 0.25), 0 4px 16px rgba(0, 0, 0, 0.1) !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; backdrop-filter: blur(10px) !important; } .gradio-markdown * { color: white !important; } .gradio-markdown h1 { color: white !important; font-size: 2.2em !important; font-weight: 800 !important; text-align: center !important; margin: 0 0 25px 0 !important; text-shadow: 0 0 15px rgba(255, 255, 255, 0.3) !important; border-bottom: 3px solid rgba(255, 255, 255, 0.3) !important; padding-bottom: 15px !important; } .gradio-markdown h2 { color: white !important; font-size: 1.6em !important; font-weight: 700 !important; margin: 25px 0 15px 0 !important; padding: 12px 20px !important; background: rgba(255, 255, 255, 0.1) !important; border-radius: 12px !important; border-left: 5px solid rgba(255, 255, 255, 0.3) !important; } .gradio-markdown h3 { color: white !important; font-size: 1.3em !important; font-weight: 650 !important; margin: 20px 0 12px 0 !important; } .gradio-markdown p { color: white !important; font-size: 1.1em !important; line-height: 1.8 !important; font-weight: 500 !important; margin: 12px 0 !important; text-align: left !important; opacity: 0.95 !important; } .gradio-markdown ul, .gradio-markdown ol { color: white !important; font-size: 1.1em !important; line-height: 1.8 !important; font-weight: 500 !important; margin: 15px 0 !important; padding-left: 25px !important; } .gradio-markdown li { color: white !important; font-size: 1.1em !important; font-weight: 500 !important; margin: 8px 0 !important; padding: 5px 0 !important; } .gradio-markdown strong { color: white !important; font-weight: 700 !important; text-shadow: 0 0 5px rgba(255, 255, 255, 0.3) !important; } .gradio-markdown em { color: white !important; font-style: italic !important; font-weight: 600 !important; } /* Highlight cho footer của báo cáo */ .gradio-markdown hr { border: 0 !important; height: 2px !important; background: linear-gradient(90deg, transparent, rgba(255, 255, 255, 0.5), transparent) !important; margin: 25px 0 15px 0 !important; } /* Style cho phần footer AI */ .gradio-markdown p:last-child { text-align: center !important; font-style: italic !important; color: white !important; font-size: 1.1em !important; margin-top: 20px !important; padding: 15px !important; background: rgba(255, 255, 255, 0.1) !important; border-radius: 10px !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; opacity: 0.9 !important; } /* Styling cho div footer trong markdown */ .gradio-markdown div { color: white !important; font-size: 1.1em !important; } /* Enhanced styling for better readability */ .gradio-markdown h3 { color: white !important; font-size: 1.5em !important; font-weight: 700 !important; margin: 25px 0 15px 0 !important; text-shadow: 0 0 10px rgba(255,255,255,0.3) !important; } .gradio-markdown h4 { color: white !important; font-size: 1.2em !important; font-weight: 650 !important; margin: 20px 0 12px 0 !important; } /* Improve list styling */ .gradio-markdown ul li { color: white !important; font-size: 1.15em !important; font-weight: 500 !important; margin: 10px 0 !important; padding: 8px 0 !important; line-height: 1.6 !important; } /* Glass morphism effects for modern look */ .glass-card { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; color: white !important; backdrop-filter: blur(15px) !important; border: 1px solid rgba(255, 255, 255, 0.2) !important; border-radius: 20px !important; padding: 20px !important; margin: 15px 0 !important; box-shadow: 0 8px 32px rgba(12, 83, 135, 0.25), 0 4px 16px rgba(0, 0, 0, 0.1) !important; } .glass-card * { color: white !important; } /* Remove extra borders and nested containers - CLEAN UP */ .info-card .gradio-group, .info-card .gradio-column, .info-card .gradio-row, .feature-box .gradio-group, .feature-box .gradio-column, .feature-box .gradio-row, .prediction-box .gradio-group, .prediction-box .gradio-column, .prediction-box .gradio-row, .plot-container .gradio-group, .plot-container .gradio-column, .plot-container .gradio-row, .glass-card .gradio-group, .glass-card .gradio-column, .glass-card .gradio-row { background: transparent !important; border: none !important; box-shadow: none !important; padding: 0 !important; margin: 0 !important; outline: none !important; } /* Simplify neon border effect - NO OVERLAPPING with other styles */ .neon-border { box-shadow: 0 0 15px rgba(12, 83, 135, 0.4) !important; border: 1px solid rgba(12, 83, 135, 0.6) !important; animation: neonPulse 3s ease-in-out infinite alternate !important; } @keyframes neonPulse { 0% { box-shadow: 0 0 15px rgba(12, 83, 135, 0.3); } 100% { box-shadow: 0 0 25px rgba(12, 83, 135, 0.5); } } /* Remove double borders on components - SIMPLIFIED */ .plot-container { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; border: 1px solid rgba(255, 255, 255, 0.3) !important; border-radius: 20px !important; padding: 15px !important; margin: 15px 0 !important; box-shadow: 0 8px 32px rgba(12, 83, 135, 0.25) !important; height: 550px !important; max-height: 650px !important; min-height: 400px !important; position: relative !important; overflow: hidden !important; backdrop-filter: blur(10px) !important; } /* Remove conflicting styles from multiple classes */ .plot-container.glass-card, .plot-container.neon-border, .glass-card.neon-border { border: 1px solid rgba(255, 255, 255, 0.3) !important; box-shadow: 0 8px 32px rgba(12, 83, 135, 0.25) !important; } /* Remove all conflicting neon border animations */ .neon-border { box-shadow: 0 0 5px rgba(12, 83, 135, 0.3), 0 0 10px rgba(12, 83, 135, 0.3), 0 0 15px rgba(12, 83, 135, 0.3), 0 0 20px rgba(12, 83, 135, 0.2), inset 0 0 5px rgba(255, 255, 255, 0.1); border: 1px solid rgba(12, 83, 135, 0.4); animation: neonPulse 2s ease-in-out infinite alternate; } /* REMOVE - this was causing duplication */ /* Modern button styling */ .btn-modern { background: linear-gradient(90deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%); border: none; border-radius: 16px; padding: 16px 32px; color: white; font-weight: 700; text-transform: uppercase; letter-spacing: 1.2px; box-shadow: 0 8px 25px rgba(12, 83, 135, 0.3), 0 3px 8px rgba(0, 0, 0, 0.1); transition: all 0.4s cubic-bezier(0.4, 0, 0.2, 1); position: relative; overflow: hidden; font-size: 0.95em; } .btn-modern::before { content: ''; position: absolute; top: 0; left: -100%; width: 100%; height: 100%; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.3), transparent); transition: left 0.6s; } .btn-modern:hover { transform: translateY(-3px) scale(1.02); box-shadow: 0 15px 40px rgba(12, 83, 135, 0.4), 0 8px 16px rgba(0, 0, 0, 0.15); } .btn-modern:hover::before { left: 100%; } .btn-modern:active { transform: translateY(-1px) scale(1.01); } /* Animated background gradients */ .bg-blue-grad { background: linear-gradient(90deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%); background-size: 200% 200%; animation: gradientShift 4s ease-in-out infinite alternate; } .bg-gray-grad { background: linear-gradient(90deg, #cbd5e1 0%, #94a3b8 25%, #64748b 50%, #475569 75%, #334155 100%); background-size: 200% 200%; animation: gradientShift 4s ease-in-out infinite alternate; } @keyframes gradientShift { 0% { background-position: 0% 50%; } 100% { background-position: 100% 50%; } } /* Tech-style borders and effects */ .tech-border { position: relative; border: 2px solid transparent; border-radius: 16px; background: linear-gradient(45deg, #A9AAA9, #8194A0, #5A7E98, #33688F, #0C5387) border-box; mask: linear-gradient(#fff 0 0) padding-box, linear-gradient(#fff 0 0); mask-composite: exclude; -webkit-mask-composite: xor; transition: all 0.3s ease; } .tech-border:hover { box-shadow: 0 0 20px rgba(12, 83, 135, 0.3); } /* Floating squares background animation */ .gradio-container::before { content: ''; position: fixed; top: 0; left: 0; width: 100%; height: 100%; pointer-events: none; z-index: -1; background-image: linear-gradient(45deg, rgba(26, 95, 145, 0.03) 25%, transparent 25%), linear-gradient(-45deg, rgba(26, 95, 145, 0.03) 25%, transparent 25%), linear-gradient(45deg, transparent 75%, rgba(26, 95, 145, 0.03) 75%), linear-gradient(-45deg, transparent 75%, rgba(26, 95, 145, 0.03) 75%); background-size: 60px 60px; background-position: 0 0, 0 30px, 30px -30px, -30px 0px; animation: floatingSquares 20s linear infinite; } @keyframes floatingSquares { 0% { transform: translateY(0px) rotate(0deg); } 100% { transform: translateY(-60px) rotate(360deg); } } /* Individual floating squares */ .floating-square { position: fixed; pointer-events: none; z-index: -1; opacity: 0.1; animation: floatUpDown 15s infinite ease-in-out; } .floating-square:nth-child(1) { width: 20px; height: 20px; background: rgba(26, 95, 145, 0.2); top: 10%; left: 10%; animation-delay: 0s; animation-duration: 12s; } .floating-square:nth-child(2) { width: 15px; height: 15px; background: rgba(139, 139, 139, 0.15); top: 20%; right: 15%; animation-delay: 2s; animation-duration: 18s; } .floating-square:nth-child(3) { width: 25px; height: 25px; background: rgba(26, 95, 145, 0.1); bottom: 30%; left: 20%; animation-delay: 4s; animation-duration: 14s; } .floating-square:nth-child(4) { width: 18px; height: 18px; background: rgba(169, 170, 169, 0.12); top: 50%; right: 25%; animation-delay: 6s; animation-duration: 16s; } .floating-square:nth-child(5) { width: 22px; height: 22px; background: rgba(26, 95, 145, 0.08); bottom: 20%; right: 10%; animation-delay: 8s; animation-duration: 20s; } .floating-square:nth-child(6) { width: 16px; height: 16px; background: rgba(90, 126, 152, 0.1); top: 70%; left: 50%; animation-delay: 10s; animation-duration: 13s; } @keyframes floatUpDown { 0%, 100% { transform: translateY(0px) rotate(0deg) scale(1); opacity: 0.1; } 25% { transform: translateY(-20px) rotate(90deg) scale(1.1); opacity: 0.2; } 50% { transform: translateY(-40px) rotate(180deg) scale(0.9); opacity: 0.15; } 75% { transform: translateY(-20px) rotate(270deg) scale(1.05); opacity: 0.18; } } /* Animated gradient background */ .gradio-container::after { content: ''; position: fixed; top: 0; left: 0; width: 100%; height: 100%; pointer-events: none; z-index: -2; background: radial-gradient(circle at 20% 20%, rgba(26, 95, 145, 0.05) 0%, transparent 50%), radial-gradient(circle at 80% 80%, rgba(139, 139, 139, 0.03) 0%, transparent 50%), radial-gradient(circle at 40% 60%, rgba(26, 95, 145, 0.02) 0%, transparent 50%); animation: gradientFloat 25s ease-in-out infinite; } @keyframes gradientFloat { 0%, 100% { transform: scale(1) rotate(0deg); opacity: 0.5; } 50% { transform: scale(1.1) rotate(180deg); opacity: 0.3; } } /* Particle effect overlay */ .particle-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; pointer-events: none; z-index: -1; background-image: radial-gradient(2px 2px at 20px 30px, rgba(26, 95, 145, 0.1), transparent), radial-gradient(2px 2px at 40px 70px, rgba(139, 139, 139, 0.08), transparent), radial-gradient(1px 1px at 90px 40px, rgba(26, 95, 145, 0.06), transparent), radial-gradient(1px 1px at 130px 80px, rgba(169, 170, 169, 0.05), transparent), radial-gradient(2px 2px at 160px 30px, rgba(26, 95, 145, 0.07), transparent); background-repeat: repeat; background-size: 200px 100px, 180px 120px, 220px 90px, 190px 110px, 210px 95px; animation: particleDrift 30s linear infinite; } @keyframes particleDrift { 0% { transform: translateX(0) translateY(0); } 100% { transform: translateX(-200px) translateY(-100px); } } /* Additional geometric shapes */ .geometric-shapes { position: fixed; top: 0; left: 0; width: 100%; height: 100%; pointer-events: none; z-index: -1; overflow: hidden; } .shape { position: absolute; opacity: 0.08; animation: float 20s infinite ease-in-out; } .shape.triangle { width: 0; height: 0; border-left: 10px solid transparent; border-right: 10px solid transparent; border-bottom: 20px solid rgba(26, 95, 145, 0.15); top: 15%; left: 80%; animation-delay: 1s; animation-duration: 25s; } .shape.circle { width: 30px; height: 30px; border-radius: 50%; background: rgba(139, 139, 139, 0.1); top: 60%; left: 15%; animation-delay: 3s; animation-duration: 18s; } .shape.hexagon { width: 20px; height: 11.55px; background: rgba(26, 95, 145, 0.12); position: relative; top: 40%; right: 5%; animation-delay: 5s; animation-duration: 22s; } .shape.hexagon:before, .shape.hexagon:after { content: ""; position: absolute; width: 0; border-left: 10px solid transparent; border-right: 10px solid transparent; } .shape.hexagon:before { bottom: 100%; border-bottom: 5.77px solid rgba(26, 95, 145, 0.12); } .shape.hexagon:after { top: 100%; border-top: 5.77px solid rgba(26, 95, 145, 0.12); } .shape.diamond { width: 15px; height: 15px; background: rgba(169, 170, 169, 0.1); transform: rotate(45deg); top: 75%; left: 70%; animation-delay: 7s; animation-duration: 16s; } @keyframes float { 0%, 100% { transform: translateY(0px) rotate(0deg); opacity: 0.08; } 25% { transform: translateY(-30px) rotate(90deg); opacity: 0.15; } 50% { transform: translateY(-60px) rotate(180deg); opacity: 0.12; } 75% { transform: translateY(-30px) rotate(270deg); opacity: 0.18; } } /* Animated grid overlay */ .grid-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; pointer-events: none; z-index: -2; background-image: linear-gradient(rgba(26, 95, 145, 0.03) 1px, transparent 1px), linear-gradient(90deg, rgba(26, 95, 145, 0.03) 1px, transparent 1px); background-size: 50px 50px; animation: gridMove 40s linear infinite; } @keyframes gridMove { 0% { transform: translate(0, 0); } 100% { transform: translate(50px, 50px); } } /* Wave animation for bottom */ .wave-container { position: fixed; bottom: 0; left: 0; width: 100%; height: 100px; pointer-events: none; z-index: -1; overflow: hidden; } .wave { position: absolute; bottom: 0; left: 0; width: 200%; height: 100px; background: linear-gradient(90deg, rgba(26, 95, 145, 0.03) 0%, rgba(139, 139, 139, 0.02) 50%, rgba(26, 95, 145, 0.03) 100%); border-radius: 100% 100% 0 0; animation: waveAnimation 15s infinite linear; } .wave:nth-child(2) { animation-delay: -5s; animation-duration: 20s; opacity: 0.5; height: 80px; } .wave:nth-child(3) { animation-delay: -10s; animation-duration: 25s; opacity: 0.3; height: 60px; } @keyframes waveAnimation { 0% { transform: translateX(-50%) rotate(0deg); } 100% { transform: translateX(-50%) rotate(360deg); } } /* Shooting stars effect */ .shooting-star { position: fixed; top: 20%; right: -10px; width: 2px; height: 2px; background: rgba(26, 95, 145, 0.8); border-radius: 50%; pointer-events: none; z-index: -1; animation: shootingStar 8s linear infinite; } .shooting-star::after { content: ''; position: absolute; top: 0; right: 0; height: 2px; width: 50px; background: linear-gradient(90deg, rgba(26, 95, 145, 0.8), transparent); } .shooting-star:nth-child(1) { animation-delay: 0s; top: 15%; } .shooting-star:nth-child(2) { animation-delay: 3s; top: 25%; animation-duration: 12s; } .shooting-star:nth-child(3) { animation-delay: 6s; top: 35%; animation-duration: 10s; } @keyframes shootingStar { 0% { transform: translateX(0) translateY(0); opacity: 1; } 70% { opacity: 1; } 100% { transform: translateX(-100vw) translateY(50px); opacity: 0; } } /* Holographic effect */ .holographic { background: linear-gradient(45deg, #A9AAA9 0%, #8194A0 20%, #5A7E98 40%, #33688F 60%, #0C5387 80%, #A9AAA9 100%); background-size: 400% 400%; animation: hologram 6s ease-in-out infinite; } @keyframes hologram { 0%, 100% { background-position: 0% 50%; } 50% { background-position: 100% 50%; } } /* Floating animation for cards */ @keyframes float { 0%, 100% { transform: translateY(0px); } 50% { transform: translateY(-5px); } } .floating { animation: float 3s ease-in-out infinite; } /* Modern card design - SINGLE DEFINITION */ .modern-card { background: linear-gradient(135deg, #A9AAA9 0%, #8194A0 25%, #5A7E98 50%, #33688F 75%, #0C5387 100%) !important; color: white !important; backdrop-filter: blur(20px) !important; border: 1px solid rgba(255, 255, 255, 0.3) !important; border-radius: 20px !important; padding: 25px !important; margin: 15px 0 !important; box-shadow: 0 8px 25px rgba(12, 83, 135, 0.25) !important; transition: all 0.4s cubic-bezier(0.4, 0, 0.2, 1) !important; position: relative !important; overflow: hidden !important; } .modern-card * { color: white !important; } .modern-card::before { content: '' !important; position: absolute !important; top: 0 !important; left: 0 !important; right: 0 !important; height: 1px !important; background: linear-gradient(90deg, transparent, rgba(255,255,255,0.8), transparent) !important; } .modern-card:hover { transform: translateY(-5px) scale(1.02) !important; box-shadow: 0 15px 35px rgba(12, 83, 135, 0.35) !important; } /* Remove duplicate info card styling */ /* Info card styling already handled in unified section above */ /* Loading spinner enhancement - cải thiện animation */ .loading-spinner { border: 4px solid rgba(255, 255, 255, 0.1); border-top: 4px solid #00ff88; border-right: 4px solid #0099ff; border-radius: 50%; width: 60px; height: 60px; animation: spin 1.2s linear infinite, pulse 2s ease-in-out infinite; margin: 0 auto 25px; box-shadow: 0 0 30px rgba(0, 255, 136, 0.4), 0 0 60px rgba(0, 153, 255, 0.2); } @keyframes spin { 0% { transform: rotate(0deg); } 100% { transform: rotate(360deg); } } @keyframes pulse { 0%, 100% { transform: scale(1); box-shadow: 0 0 30px rgba(0, 255, 136, 0.4); } 50% { transform: scale(1.05); box-shadow: 0 0 40px rgba(0, 255, 136, 0.6), 0 0 80px rgba(0, 153, 255, 0.3); } } /* Cải thiện progress bar */ .progress-bar { background: rgba(255, 255, 255, 0.1); border-radius: 15px; height: 8px; margin: 15px 0; overflow: hidden; box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.2); } .progress-fill { background: linear-gradient(90deg, #00ff88, #0099ff, #00ff88); height: 100%; border-radius: 15px; transition: width 0.5s ease-out; animation: shimmer 2s infinite; box-shadow: 0 0 10px rgba(0, 255, 136, 0.5); } @keyframes shimmer { 0% { background-position: -200% 0; } 100% { background-position: 200% 0; } } /* Dropdown info text styling - make text white using #FFFFFF */ .gradio-dropdown .info, .gradio-dropdown .gr-form-info, .gradio-dropdown .svelte-1gfkn6j, label[data-testid="block-info"], .gr-form .info, .gradio-container .info, .gradio-container [data-testid="block-info"], .gradio-container .gr-form .info, .gradio-container .gr-form-info, .gradio-container .svelte-1gfkn6j, .gradio-container span.info, .gradio-container p.info, .gradio-container div.info, .gradio-container .block-info, .gradio-app .info, .gradio-app [data-testid="block-info"], .gradio-app .gr-form-info, .gradio-app .svelte-1gfkn6j { color: #FFFFFF !important; opacity: 1 !important; } /* More specific targeting for dropdown info */ .gr-dropdown .info, .gr-dropdown [data-testid="block-info"], .gr-textbox .info, .gr-textbox [data-testid="block-info"] { color: #FFFFFF !important; opacity: 1 !important; } /* Global override for all info elements */ * .info { color: #FFFFFF !important; } *[data-testid="block-info"] { color: #FFFFFF !important; } /* Specific class for dropdown with white info text */ .white-info-dropdown .info, .white-info-dropdown [data-testid="block-info"], .white-info-dropdown span, .white-info-dropdown p { color: #FFFFFF !important; opacity: 1 !important; } /* Force all Gradio info text to be white */ .gradio-container * { --block-info-text-color: #FFFFFF !important; } .gradio-container .block-info, .gradio-container .info-text, .gradio-container .description { color: #FFFFFF !important; } /* Specific targeting for spans with data-testid="block-info" */ span[data-testid="block-info"], div[data-testid="block-info"], label[data-testid="block-info"], .svelte-g2oxp3.has-info { color: #FFFFFF !important; background: transparent !important; text-shadow: 0 0 5px rgba(0, 0, 0, 0.5) !important; } /* Target the specific class mentioned */ .svelte-g2oxp3 { color: #FFFFFF !important; } /* Radio button and form labels improvements */ .gradio-radio label, .gradio-radio span, .gradio-dropdown label, .gradio-dropdown span { color: #FFFFFF !important; font-weight: 600 !important; text-shadow: 0 0 3px rgba(0, 0, 0, 0.3) !important; } /* Form component styling */ .gradio-radio, .gradio-dropdown { background: transparent !important; border: none !important; padding: 10px !important; } /* Remove extra backgrounds on form components */ .gradio-radio .form, .gradio-dropdown .form, .gradio-button .form { background: transparent !important; border: none !important; box-shadow: none !important; } /* Clean up nested containers */ .gradio-column > div, .gradio-row > div { background: transparent !important; border: none !important; box-shadow: none !important; } """ # ---------- MODULE STREAMING THỰC SỰ ---------- """ Hiệu ứng 'AI đang gõ' cho báo cáo CEO: - tao_bao_cao_noi_dung: tạo toàn văn báo cáo (markdown, thuần Việt) - tao_bao_cao_streaming: generator stream từng cụm có con trỏ nhấp nháy - tao_streaming_text: alias dùng cho tương thích ngược - xu_ly_du_doan_voi_loading: trả về (fig, tóm tắt, streamer hoặc full text) - bat_dau_loading / tinh_toan_ket_qua: workflow loading + stream - cap_nhat_thong_tin_case, tao_csv_gia_lap: tiện ích trình bày demo Lưu ý: - Mặc định trả về generator để front-end cập nhật liên tục. - Nếu framework của bạn không hỗ trợ generator, gọi với streaming=False để nhận về chuỗi đầy đủ rồi render một lần. """ # ---------- TIỆN ÍCH NHỎ ---------- def tao_loading_html_streaming(thong_diep: str = "🔎 Đang xử lý", do_mo: float = 0.15) -> str: """Khối HTML loading gọn, dễ nhúng.""" return f"""
Sản phẩm: {thong_tin.get('san_pham', '')}
Mô tả: {thong_tin.get('mo_ta', '')}
{chi_tiet_buoc}
🤖 FoxAI đang xử lý {pct}% • Vui lòng chờ trong giây lát...
🎯 Chọn SKU (Thủ Đô / Thăng Long) và nhấn "Tạo Dự Báo AI"
Hệ thống sẽ tính xu hướng, tồn an toàn, ROP và cảnh báo kênh GT/MT
{tong_don_vi:,} cây
{trung_binh:,.0f} cây
{dinh:,} cây
{ty_le_tang_truong:+.1f}%
📅 Chu Kỳ Dự Báo: {chu_ky} | Mức dịch vụ mục tiêu: {muc_dich_vu:.0%} | LT: {lead_time_ngay} ngày
⚡ Phân tích dữ liệu • 🧠 Machine Learning • 📊 Tạo insights • 🎯 Chiến lược AI
Phân tích thông minh với công nghệ FoxAI
Chọn sản phẩm, chu kỳ dự báo và nhận phân tích AI chi tiết
Trạng thái kết nối cơ sở dữ liệu: Ổn định
Định dạng: Time series data với cột thời gian và doanh thu
Cập nhật: Tự động sync theo mặt hàng được chọn
Những công nghệ tiên tiến đang được nghiên cứu và phát triển
Kết hợp nhiều kiến trúc AI (RNN, Transformer, Prophet, XGBoost...) để tăng độ chính xác và ổn định của dự báo.
Mô hình tự động học lại khi có thay đổi bất thường về nhu cầu (ví dụ mùa vụ, sự kiện kinh tế, khuyến mãi).
Giải thích chi tiết tại sao hệ thống đưa ra dự báo cụ thể, hiển thị các yếu tố tác động chính (giá, mùa vụ, đối thủ cạnh tranh).
Cập nhật dự báo liên tục từ dòng dữ liệu trực tiếp, kèm cảnh báo khi có tín hiệu bất thường.
Cho phép người dùng giả lập các kịch bản (tăng giá, thay đổi kênh phân phối, biến động thị trường) và xem tác động dự báo ngay.
Dự báo chi tiết tới từng SKU, từng khu vực, thậm chí từng khách hàng trọng điểm.
Tự động sinh báo cáo phân tích bằng ngôn ngữ tự nhiên (Natural Language Generation) → người dùng không cần đọc chart quá nhiều.
Tạo insights dạng văn bản từ dữ liệu
Báo cáo bằng giọng nói tự nhiên
Hỏi đáp trực tiếp với AI
Gửi tóm tắt qua email tự động
🦊 Powered by FoxAi • 🇻🇳 Made in Vietnam • 🤖 Advanced AI
YOUR TRUSTED AI PARTNER