Spaces:
Sleeping
Sleeping
| # -*- 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""" | |
| <div class="loading-card" style="padding:14px;border-radius:10px;background:rgba(255,255,255,{do_mo});border:1px solid rgba(255,255,255,0.25);"> | |
| <div style="display:flex;align-items:center;gap:10px;"> | |
| <div class="spinner" style="width:16px;height:16px;border:3px solid #ddd;border-top-color:#1A5F91;border-radius:50%;animation:spin 1s linear infinite;"></div> | |
| <strong>{thong_diep}</strong> | |
| </div> | |
| </div> | |
| <style> | |
| @keyframes spin {{ from {{transform:rotate(0deg)}} to {{transform:rotate(360deg)}} }} | |
| </style> | |
| """ | |
| def _pause_for_token(token: str, toc_do: float, he_so_dau_cau: float = 1.8) -> None: | |
| """Dừng lâu hơn sau dấu câu/kết dòng để cảm giác tự nhiên.""" | |
| if re.search(r"[.!?…:;。!?]$", token) or token.endswith("\n"): | |
| time.sleep(toc_do * he_so_dau_cau) | |
| else: | |
| time.sleep(toc_do) | |
| def _chunk_by_words(text: str, kich_thuoc: int = 3) -> Iterable[str]: | |
| """Chia text theo cụm ~N từ, giữ dấu câu.""" | |
| words = re.findall(r"\S+\s*", text) | |
| buf = [] | |
| for w in words: | |
| buf.append(w) | |
| if len(buf) >= kich_thuoc or w.endswith("\n"): | |
| yield "".join(buf) | |
| buf = [] | |
| if buf: | |
| yield "".join(buf) | |
| # ---------- NỘI DUNG BÁO CÁO ---------- | |
| def tao_bao_cao_noi_dung_streaming(ten_case: str) -> str: | |
| """ | |
| Tạo toàn văn báo cáo (markdown) từ TRUONG_HOP_DEMO[ten_case]. | |
| Nội dung thuần Việt, dễ đọc cho CEO. | |
| """ | |
| thong_tin_case = TRUONG_HOP_DEMO[ten_case] | |
| gd = thong_tin_case["thong_tin_giam_doc"] | |
| # Khối kết luận được Việt hóa gọn, tránh trộn Anh-Việt. | |
| ket_luan = ( | |
| "• Mức độ ưu tiên: cao\n" | |
| "• Thời điểm triển khai: ngay\n" | |
| "• Hiệu quả kỳ vọng: cải thiện 15–25% trong 6 tháng (ước tính)\n" | |
| "• Mức rủi ro: thấp nếu theo dõi chỉ số đều tay\n" | |
| ) | |
| noi_dung = ( | |
| f"# 🎯 KẾ HOẠCH HÀNH ĐỘNG CHO {ten_case.upper()}\n\n" | |
| "## 📈 Tóm tắt nhanh\n\n" | |
| f"{gd['tom_tat']}\n\n" | |
| "## 📊 Diễn giải theo biểu đồ\n\n" | |
| f"{gd['chi_tiet']}\n\n" | |
| "## 🚀 Việc cần làm ngay\n\n" | |
| f"{gd['hanh_dong']}\n\n" | |
| "## ✅ Kết luận\n\n" | |
| f"{ket_luan}\n" | |
| "---\n\n" | |
| "<div style='text-align:center;font-weight:600;padding:12px;" | |
| "background:rgba(255,255,255,0.12);border:1px solid rgba(255,255,255,0.2);" | |
| "border-radius:10px;'>🤖 FoxAI Business Intelligence • ⚡ Cập nhật theo thời gian thực</div>" | |
| ) | |
| return noi_dung | |
| # ---------- STREAMING ---------- | |
| def tao_bao_cao_streaming_moi( | |
| ten_case: str, | |
| kieu: str = "word", # "word" | "sentence" | "char" | |
| toc_do: float = 0.03, # giây mỗi cụm | |
| con_tro: bool = True, # hiển thị con trỏ nhấp nháy ▌ | |
| cum_tu: int = 3 # số từ mỗi cụm nếu kieu="word" | |
| ) -> Generator[str, None, None]: | |
| """ | |
| Generator gõ "từng cụm" như AI đang viết. | |
| Trả về các lần cập nhật tích lũy (phù hợp vùng text duy nhất trên UI). | |
| """ | |
| full_text = tao_bao_cao_noi_dung_streaming(ten_case) | |
| accumulated = "" | |
| cursor_on = True | |
| if kieu == "char": | |
| tokens = list(full_text) | |
| elif kieu == "sentence": | |
| # Chia theo câu, vẫn đảm bảo xuống dòng tự nhiên | |
| tokens = re.split(r"(?<=[.!?…:;。!?])\s+", full_text) | |
| tokens = [t + " " for t in tokens if t] | |
| else: | |
| tokens = list(_chunk_by_words(full_text, kich_thuoc=cum_tu)) | |
| for tk in tokens: | |
| accumulated += tk | |
| frame = accumulated | |
| if con_tro: | |
| frame = frame + (" ▌" if cursor_on else " ") | |
| cursor_on = not cursor_on | |
| yield frame | |
| _pause_for_token(tk, toc_do, he_so_dau_cau=1.8) | |
| # Khung cuối cùng tắt con trỏ để đỡ nhấp nháy sau khi xong | |
| if con_tro: | |
| yield accumulated | |
| def tao_streaming_text_moi( | |
| ten_case: str, | |
| kieu: str = "word", | |
| toc_do: float = 0.03, | |
| con_tro: bool = True, | |
| cum_tu: int = 3 | |
| ) -> Generator[str, None, None]: | |
| """Alias để tương thích ngược.""" | |
| return tao_bao_cao_streaming_moi( | |
| ten_case, kieu=kieu, toc_do=toc_do, con_tro=con_tro, cum_tu=cum_tu | |
| ) | |
| # ---------- WORKFLOW LOADING + STREAM ---------- | |
| def xu_ly_du_doan_voi_loading_moi( | |
| lua_chon_case: Optional[str], | |
| chu_ky: str, | |
| che_do_xem: str, | |
| streaming: bool = True, | |
| kieu: str = "word", | |
| toc_do: float = 0.03, | |
| cum_tu: int = 3 | |
| ): | |
| """ | |
| Xử lý dự đoán + chuẩn bị nội dung báo cáo. | |
| - streaming=True: trả về generator để UI cập nhật dần | |
| - streaming=False: trả về chuỗi đầy đủ | |
| """ | |
| if lua_chon_case is None: | |
| fig = tao_bieu_do_mac_dinh() | |
| tom_tat = tao_tom_tat_mac_dinh() | |
| bao_cao = "# 🎯 KẾ HOẠCH HÀNH ĐỘNG\n\nChọn mặt hàng để xem phân tích." | |
| return fig, tom_tat, bao_cao | |
| # Delay nhẹ để cảm giác "đang tính" | |
| time.sleep(0.25) | |
| bieu_do = tao_bieu_do_du_doan(lua_chon_case, chu_ky, che_do_xem) | |
| time.sleep(0.15) | |
| tom_tat = tao_thong_tin_tom_tat(lua_chon_case, chu_ky, che_do_xem) | |
| time.sleep(0.20) | |
| if streaming: | |
| bao_cao_gen = tao_bao_cao_streaming_moi( | |
| lua_chon_case, kieu=kieu, toc_do=toc_do, cum_tu=cum_tu | |
| ) | |
| return bieu_do, tom_tat, bao_cao_gen | |
| else: | |
| bao_cao_full = tao_bao_cao_noi_dung_streaming(lua_chon_case) | |
| return bieu_do, tom_tat, bao_cao_full | |
| def bat_dau_loading_moi() -> Tuple[go.Figure, str, str, str]: | |
| """ | |
| Bật panel loading trước khi stream để auto-scroll trên UI. | |
| Trả về: (fig_placeholder, tom_tat_placeholder, text_placeholder, html_loading) | |
| """ | |
| html = tao_loading_html_streaming("🔎 Đang kiểm tra dữ liệu", 0.12) | |
| return tao_bieu_do_mac_dinh(), tao_tom_tat_mac_dinh(), "## ⏳ Đang chuẩn bị...", html | |
| def tinh_toan_ket_qua_moi( | |
| lua_chon_case: str, | |
| chu_ky: str, | |
| che_do_xem: str, | |
| streaming: bool = True | |
| ) -> Tuple[go.Figure, str, Iterable[str] | str, str]: | |
| """ | |
| Tính toán kết quả thật sự (demo data). | |
| Trả về: (figure, html_tóm_tắt, báo_cáo_stream hoặc toàn_văn, html_info) | |
| """ | |
| bieu_do, tom_tat, bao_cao = xu_ly_du_doan_voi_loading_moi( | |
| lua_chon_case, chu_ky, che_do_xem, streaming=streaming | |
| ) | |
| csv_info = f""" | |
| <div class='info-card'> | |
| ✅ <strong>Đã phân tích dữ liệu chuỗi thời gian</strong><br/> | |
| 📦 <strong>SKU:</strong> {lua_chon_case}<br/> | |
| 📅 <strong>Chu kỳ:</strong> {chu_ky} | 🔍 <strong>Chế độ:</strong> {che_do_xem} | |
| </div> | |
| """.strip() | |
| return bieu_do, tom_tat, bao_cao, csv_info | |
| # ---------- TIỆN ÍCH DEMO ---------- | |
| def cap_nhat_thong_tin_case_moi(lua_chon_case: str) -> str: | |
| """Thẻ thông tin ngắn về case đã chọn.""" | |
| if lua_chon_case in TRUONG_HOP_DEMO: | |
| thong_tin = TRUONG_HOP_DEMO[lua_chon_case] | |
| return f""" | |
| <div class='info-card'> | |
| <h4>📋 Tình huống: {lua_chon_case}</h4> | |
| <p><strong>Sản phẩm:</strong> {thong_tin.get('san_pham', '')}</p> | |
| <p><strong>Mô tả:</strong> {thong_tin.get('mo_ta', '')}</p> | |
| </div> | |
| """ | |
| return "" | |
| def tao_csv_gia_lap_moi(ten_case: str, chu_ky: str, che_do_xem: str): | |
| """ | |
| Tạo DataFrame giả lập như CSV thật để demo. | |
| Trả về: (df, ten_cot_thoi_gian, ten_cot_gia_tri) | |
| """ | |
| thong_tin_case = TRUONG_HOP_DEMO[ten_case] | |
| nhan_thoi_gian, don_vi_ban = tao_du_lieu_demo(thong_tin_case, chu_ky, che_do_xem) | |
| san_pham = thong_tin_case.get("san_pham", "") | |
| if che_do_xem == "ngày": | |
| dates = [ | |
| (datetime.now() - timedelta(days=len(nhan_thoi_gian) - i - 1)).strftime("%Y-%m-%d") | |
| for i in range(len(nhan_thoi_gian)) | |
| ] | |
| df_fake = pd.DataFrame( | |
| {"Ngay": dates, "So_Cay": don_vi_ban, "San_Pham": [san_pham] * len(don_vi_ban)} | |
| ) | |
| return df_fake, "Ngay", "So_Cay" | |
| if che_do_xem == "tháng": # theo tuần | |
| dates = [ | |
| (datetime.now() - timedelta(weeks=len(nhan_thoi_gian) - i - 1)).strftime("%Y-%m-%d") | |
| for i in range(len(nhan_thoi_gian)) | |
| ] | |
| df_fake = pd.DataFrame( | |
| {"Tuan": dates, "So_Cay": don_vi_ban, "San_Pham": [san_pham] * len(don_vi_ban)} | |
| ) | |
| return df_fake, "Tuan", "So_Cay" | |
| # năm = theo tháng | |
| dates = [ | |
| (datetime.now() - timedelta(days=30 * (len(nhan_thoi_gian) - i - 1))).strftime("%Y-%m-%d") | |
| for i in range(len(nhan_thoi_gian)) | |
| ] | |
| df_fake = pd.DataFrame( | |
| {"Thang": dates, "So_Cay": don_vi_ban, "San_Pham": [san_pham] * len(don_vi_ban)} | |
| ) | |
| return df_fake, "Thang", "So_Cay" | |
| # ---------- KẾT THÚC MODULE STREAMING ---------- | |
| def tao_loading_html(thong_diep: str = "🔎 Đang xử lý", ti_le: float = 0.15) -> str: | |
| """Tạo HTML loading với thanh tiến độ theo tỷ lệ [0..1].""" | |
| pct = max(0, min(100, int(ti_le * 100))) | |
| # Thêm các thông điệp chi tiết dựa trên tiến độ | |
| chi_tiet_buoc = "" | |
| if ti_le <= 0.25: | |
| chi_tiet_buoc = "📡 Kết nối với AI Engine • 🔍 Phân tích cấu trúc dữ liệu" | |
| elif ti_le <= 0.50: | |
| chi_tiet_buoc = "🧹 Làm sạch và chuẩn hóa dữ liệu • 📈 Xác định xu hướng" | |
| elif ti_le <= 0.75: | |
| chi_tiet_buoc = "🧠 Huấn luyện mô hình ML • ⚡ Tối ưu hóa thuật toán" | |
| else: | |
| chi_tiet_buoc = "📊 Tạo insights • 🎯 Phân tích chiến lược • 📝 Hoàn thiện báo cáo" | |
| return f""" | |
| <div class="loading-container"> | |
| <div class="loading-spinner"></div> | |
| <h3 style="color: white; margin: 0 0 10px 0; font-size: 18px; font-weight: 600;">{thong_diep}...</h3> | |
| <div class="progress-bar"> | |
| <div class="progress-fill" style="width:{pct}%"></div> | |
| </div> | |
| <p style="color: rgba(255,255,255,0.9); margin: 15px 0 5px 0; font-size: 13px;"> | |
| {chi_tiet_buoc} | |
| </p> | |
| <p style="color: rgba(255,255,255,0.7); margin: 5px 0; font-size: 12px;"> | |
| 🤖 FoxAI đang xử lý {pct}% • Vui lòng chờ trong giây lát... | |
| </p> | |
| </div> | |
| """ | |
| def hien_loading_tung_buoc(): | |
| """Generator: stream 6 bước loading trong khoảng 5–8 giây để cảm giác thực tế hơn.""" | |
| tong_thoi_gian = random.uniform(5.0, 8.0) # Tăng thời gian loading | |
| buoc = [ | |
| ("� Đang khởi tạo hệ thống AI", 0.10), | |
| ("📊 Đang phân tích dữ liệu", 0.25), | |
| ("🧹 Đang tiến hành làm sạch dữ liệu", 0.45), | |
| ("🧠 Đang huấn luyện mô hình ML", 0.65), | |
| ("⚡ Đang tối ưu thuật toán", 0.85), | |
| ("📝 Đang tạo báo cáo với AI", 0.95), | |
| ] | |
| # Thời gian ngẫu nhiên cho mỗi bước để tự nhiên hơn | |
| for i, (thong_diep, ti_le) in enumerate(buoc): | |
| yield tao_loading_html(thong_diep, ti_le) | |
| # Thời gian delay ngẫu nhiên cho mỗi bước - tăng thời gian để thực tế hơn | |
| if i < len(buoc) - 1: # Không delay ở bước cuối | |
| if i == 3: # Bước huấn luyện ML - delay lâu hơn | |
| buoc_delay = random.uniform(1.5, 2.5) | |
| elif i == 4: # Bước tối ưu - delay vừa phải | |
| buoc_delay = random.uniform(1.0, 1.8) | |
| else: | |
| buoc_delay = random.uniform(0.8, 1.4) | |
| time.sleep(buoc_delay) | |
| # Delay thêm một chút trước khi hoàn tất để user thấy tiến trình | |
| time.sleep(0.8) | |
| yield tao_loading_html("✅ Hoàn tất phân tích AI", 1.0) | |
| # Delay cuối để user thấy trạng thái hoàn tất rõ ràng | |
| time.sleep(0.6) | |
| def doc_csv(file_obj): | |
| """Đọc CSV, trả preview + gợi ý cột thời gian/giá trị + state df.""" | |
| if file_obj is None: | |
| return ( | |
| gr.update(visible=False), | |
| gr.update(choices=[], value=None), | |
| gr.update(choices=[], value=None), | |
| None, | |
| "<div class='info-card'>📂 Chưa chọn CSV.</div>", | |
| ) | |
| try: | |
| df = pd.read_csv(file_obj.name) | |
| sample = df.head(100) | |
| cols = list(df.columns) | |
| # Đoán cột thời gian và giá trị | |
| guess_date = next( | |
| (c for c in cols if any(k in c.lower() for k in ["date", "time", "ngay", "thoi"])), | |
| cols[0] if cols else None, | |
| ) | |
| numeric_cols = [c for c in cols if pd.api.types.is_numeric_dtype(df[c])] | |
| guess_value = numeric_cols[0] if numeric_cols else (cols[1] if len(cols) > 1 else cols[0]) | |
| thong_bao = f"<div class='info-card'>✅ CSV đã tải: {len(df):,} dòng, {len(cols)} cột.</div>" | |
| return ( | |
| gr.update(visible=True, value=sample), | |
| gr.update(choices=cols, value=guess_date), | |
| gr.update(choices=cols, value=guess_value), | |
| df, | |
| thong_bao, | |
| ) | |
| except Exception as exc: | |
| msg = f"<div class='info-card'>❌ Lỗi đọc CSV: {exc}</div>" | |
| return ( | |
| gr.update(visible=False), | |
| gr.update(choices=[], value=None), | |
| gr.update(choices=[], value=None), | |
| None, | |
| msg, | |
| ) | |
| def tao_chuoi_tu_csv( | |
| df_user: pd.DataFrame, | |
| cot_ngay: str, | |
| cot_gia_tri: str, | |
| che_do_xem: str, | |
| ): | |
| """ | |
| Từ CSV -> (nhãn thời gian, giá trị) theo chế độ xem: | |
| 'ngày' = theo ngày, 'tháng' = theo tuần, 'năm' = theo tháng. | |
| """ | |
| if df_user is None or not cot_ngay or not cot_gia_tri: | |
| return None, None | |
| df = df_user.copy() | |
| df[cot_ngay] = pd.to_datetime(df[cot_ngay], errors="coerce") | |
| df = df.dropna(subset=[cot_ngay, cot_gia_tri]) | |
| df = df.sort_values(cot_ngay) | |
| if df.empty: | |
| return None, None | |
| if che_do_xem == "ngày": | |
| ser = df.groupby(df[cot_ngay].dt.date)[cot_gia_tri].sum() | |
| nhan = [d.strftime("%d/%m") for d in pd.to_datetime(ser.index)] | |
| elif che_do_xem == "tháng": # theo tuần | |
| ser = df.set_index(cot_ngay)[cot_gia_tri].resample("W").sum() | |
| nhan = [idx.strftime("Tuần %W/%Y") for idx in ser.index] | |
| else: # 'năm' = theo tháng | |
| ser = df.set_index(cot_ngay)[cot_gia_tri].resample("M").sum() | |
| nhan = [idx.strftime("%m/%Y") for idx in ser.index] | |
| gia_tri = ser.astype(float).tolist() | |
| if len(gia_tri) < 3: | |
| # Dọn cho gọn, biểu đồ ít hơn 3 điểm nhìn hơi thảm | |
| return None, None | |
| return nhan, gia_tri | |
| """ | |
| Dataset demo 3 SKU. Giữ nguyên ý nghĩa trường cũ, thêm trường phân tích: | |
| - gia_ban, gia_doi_thu, do_co_gian_gia: phục vụ mô phỏng giá/khuyến mãi | |
| - muc_dich_vu, lead_time_ngay: tính tồn an toàn và ROP | |
| - ty_le_kenh: phân bổ theo kênh (phần trăm) | |
| - quan_he_cross_sub: ma trận thay thế chéo giữa 3 SKU | |
| - kpi_van_hanh: chỗ điền kết quả tính toán (tồn an toàn, ROP, thiếu hàng) | |
| """ | |
| # -*- coding: utf-8 -*- | |
| """ | |
| Dataset DEMO 3 SKU. Đã chuẩn hóa chuỗi nhiều dòng cho 'thong_tin_giam_doc' | |
| để tránh lỗi cú pháp và bảo đảm 3 khóa SKU đều truy cập được. | |
| """ | |
| TRUONG_HOP_DEMO = { | |
| "Thủ Đô bao cứng (NH214/07)": { | |
| "san_pham": "Thủ Đô", | |
| "phan_loai": "bao_cung", | |
| "ma_sku": "NH214/07", | |
| "canh_bao": None, | |
| "don_vi_co_ban": 180, | |
| "don_vi_tinh": "cay/thang", | |
| "don_vi_quy_doi": {"1_cay": 10, "don_vi_con": "bao"}, | |
| "xu_huong": "on_dinh", | |
| "mo_ta": "Thủ Đô bao cứng NH214/07 phục vụ kênh GT/MT.", | |
| "kenh_ban": ["GT", "MT"], | |
| "ty_le_kenh": {"GT": 0.60, "MT": 0.40, "Online": 0.00}, | |
| "gia_ban": 100.0, | |
| "gia_doi_thu": 98.0, | |
| "do_co_gian_gia": -0.8, | |
| "muc_dich_vu": 0.95, | |
| "lead_time_ngay": 14, | |
| "quan_he_cross_sub": { | |
| "TL-COMPACT-HONG": -0.10, | |
| "TL-CB-RANG": -0.05, | |
| }, | |
| "kpi_van_hanh": { | |
| "ton_an_toan_tuan": None, | |
| "rop": None, | |
| "ngay_thieu_hang_thang": None, | |
| }, | |
| "thong_tin_giam_doc": { | |
| "tom_tat": ( | |
| "📌 Đường bán đi ngang, sóng thấp. Đây là sản phẩm giữ nhịp doanh thu ở kênh GT (~60%). " | |
| "Giá cao hơn đối thủ khoảng 2% nhưng nhu cầu không quá nhạy với giá. Nguy cơ chính là " | |
| "thiếu hàng cục bộ vào cuối tháng. Giữ mức phục vụ 95% và dự trữ tương đương 3–4 tuần bán." | |
| ), | |
| "chi_tiet": ( | |
| "• Đồ thị: dao động nhẹ, ít đỉnh bất thường; hay trồi sụt nhẹ tuần chốt tháng.\n" | |
| "• Khách hàng: trung thành, ít đổi sang biến thể khác khi trưng bày ổn.\n" | |
| "• Giá: ~100 so với đối thủ ~98; nếu tăng 1% có thể giảm ~0.8% sản lượng.\n" | |
| "• Kênh: GT 60%, MT 40%; khu vực tỉnh là chính, nhạy với mức chiết khấu tại điểm bán.\n" | |
| "• Liên đới: khi đẩy Thăng Long (Compact/Răng), Thủ Đô có thể sụt nhẹ.\n" | |
| "• Cần theo dõi: ngày thiếu hàng, tỷ lệ đơn phục vụ đủ, chênh tồn GT/MT." | |
| ), | |
| "hanh_dong": ( | |
| "• Trước ngày chốt tháng 10–14 ngày, bổ sung hàng về điểm bán đều tay; không chờ sát ngày.\n" | |
| "• Duy trì điểm đặt hàng lại; không để tồn thực tế xuống dưới mức an toàn.\n" | |
| "• Không mở khuyến mãi rộng; chỉ kích tại điểm có hiệu quả sau trừ chi phí tốt.\n" | |
| "• Trưng bày sạch, đủ mặt tiền ở GT; ưu tiên cấp hàng cho điểm bán ổn định." | |
| ), | |
| } | |
| }, | |
| "Thăng Long bao cứng Compact (cảnh báo họng)": { | |
| "san_pham": "Thăng Long", | |
| "phan_loai": "bao_cung_compact", | |
| "ma_sku": "TL-COMPACT-HONG", | |
| "canh_bao": "hong", | |
| "don_vi_co_ban": 150, | |
| "don_vi_tinh": "cay/thang", | |
| "don_vi_quy_doi": {"1_cay": 10, "don_vi_con": "bao"}, | |
| "xu_huong": "tang_nhe", | |
| "mo_ta": "Thăng Long compact, in cảnh báo họng.", | |
| "kenh_ban": ["GT", "MT", "Online"], | |
| "ty_le_kenh": {"GT": 0.50, "MT": 0.35, "Online": 0.15}, | |
| "gia_ban": 96.0, | |
| "gia_doi_thu": 97.0, | |
| "do_co_gian_gia": -1.1, | |
| "muc_dich_vu": 0.95, | |
| "lead_time_ngay": 10, | |
| "quan_he_cross_sub": { | |
| "NH214/07": -0.02, | |
| "TL-CB-RANG": -0.12, | |
| }, | |
| "kpi_van_hanh": { | |
| "ton_an_toan_tuan": None, | |
| "rop": None, | |
| "ngay_thieu_hang_thang": None, | |
| }, | |
| "thong_tin_giam_doc": { | |
| "tom_tat": ( | |
| "📈 Đường bán bò lên đều; mỗi đợt trưng bày tốt hay gói mua kèm đều tạo nhịp tăng. " | |
| "Mạnh ở đô thị, kênh MT và bán online. Giá đang thấp hơn đối thủ ~1%, nhu cầu khá nhạy giá. " | |
| "Cần tách lịch khuyến mãi với biến thể Răng để tránh khách đổi sang nhau." | |
| ), | |
| "chi_tiet": ( | |
| "• Đồ thị: xu hướng đi lên; đỉnh thường rơi vào cuối tuần/đợt lương.\n" | |
| "• Trưng bày: ở tầm mắt hoặc cạnh quầy giúp tăng bán rõ rệt; nên thử A/B vị trí.\n" | |
| "• Giá: ~96 so với 97; chỉnh giá 1% có thể đổi ~1.1% sản lượng.\n" | |
| "• Kênh: GT 50%, MT 35%, online 15%; online tạo sóng cuối tuần nhờ gói mua kèm.\n" | |
| "• Ăn lẫn nhau: khi Răng giảm giá cùng lúc, Compact bị kéo khách đáng kể.\n" | |
| "• Cần theo dõi: tỉ lệ trưng bày đạt chuẩn, số đơn từ gói mua kèm, hiệu quả khuyến mãi theo kênh." | |
| ), | |
| "hanh_dong": ( | |
| "• Chạy gói mua kèm vào cuối tuần và đợt lương; khung thời gian rõ để tạo khan hiếm.\n" | |
| "• Chọn 3 vị trí trưng bày then chốt ở MT; kiểm tra tuân thủ hàng tuần.\n" | |
| "• Giữ khoảng cách giá tối thiểu 3–5% với biến thể Răng; không khuyến mãi cùng thời điểm.\n" | |
| "• Dự trữ thêm trước cao điểm; ưu tiên cấp hàng cho MT và bán online." | |
| ), | |
| } | |
| }, | |
| "Thăng Long cảnh báo răng": { | |
| "san_pham": "Thăng Long", | |
| "phan_loai": "bao_cung", | |
| "ma_sku": "TL-CB-RANG", | |
| "canh_bao": "rang", | |
| "don_vi_co_ban": 140, | |
| "don_vi_tinh": "cay/thang", | |
| "don_vi_quy_doi": {"1_cay": 10, "don_vi_con": "bao"}, | |
| "xu_huong": "bien_dong", | |
| "mo_ta": "Thăng Long bao cứng, biến thể cảnh báo răng.", | |
| "kenh_ban": ["GT", "Chợ truyền thống"], | |
| "ty_le_kenh": {"GT": 0.80, "MT": 0.20, "Online": 0.00}, | |
| "gia_ban": 94.0, | |
| "gia_doi_thu": 95.0, | |
| "do_co_gian_gia": -1.3, | |
| "muc_dich_vu": 0.95, | |
| "lead_time_ngay": 7, | |
| "quan_he_cross_sub": { | |
| "NH214/07": -0.01, | |
| "TL-COMPACT-HONG": -0.08, | |
| }, | |
| "kpi_van_hanh": { | |
| "ton_an_toan_tuan": None, | |
| "rop": None, | |
| "ngay_thieu_hang_thang": None, | |
| }, | |
| "thong_tin_giam_doc": { | |
| "tom_tat": ( | |
| "↕️ Đường bán nhấp nhô mạnh; phụ thuộc chương trình ngắn ngày và độ phủ cửa hàng nhỏ. " | |
| "Trọng tâm GT (~80%). Giá thấp hơn đối thủ ~1% nhưng rất nhạy giá. Cần phân bổ theo " | |
| "bản đồ điểm nóng để tránh nơi thừa nơi thiếu." | |
| ), | |
| "chi_tiet": ( | |
| "• Đồ thị: biên độ cao; vừa có khuyến mãi là bật đỉnh, hết chương trình là hạ nhanh.\n" | |
| "• Giá: ~94 so với 95; chỉnh 1% có thể đổi ~1.3% sản lượng.\n" | |
| "• Kênh: GT là chủ lực; nên chia nhỏ theo phường/xã để cấp hàng đúng chỗ.\n" | |
| "• Rủi ro: đẩy hàng sát ngày chốt dễ phát sinh tồn cục bộ; khách dễ đổi sang Compact khi giá gần nhau.\n" | |
| "• Cần theo dõi: ngày thiếu hàng, tốc độ quay vòng, điểm nóng bán ra 4 tuần gần nhất." | |
| ), | |
| "hanh_dong": ( | |
| "• Chạy gói kích cầu 2–3 tuần; nếu hiệu quả thấp thì dừng sớm, tránh đốt ngân sách.\n" | |
| "• Cấp hàng dựa trên sức bán tuần; điều chuyển nhanh khỏi điểm chậm quay.\n" | |
| "• Trưng bày ngang tầm mắt, vật dụng gọn nhẹ, thay mới thường xuyên.\n" | |
| "• Không khuyến mãi trùng thời điểm với Compact; giữ khoảng giá đủ xa để hạn chế đổi sang nhau.\n" | |
| "• Lập lịch cấp hàng 2 lần/tuần ở khu vực biến động; bật cảnh báo khi tồn xuống dưới ngưỡng." | |
| ), | |
| } | |
| }, | |
| } | |
| # -*- coding: utf-8 -*- | |
| """ | |
| Các hàm hiển thị/giả lập dữ liệu theo domain hiện tại: | |
| - SKU thuốc điếu: Thủ Đô NH214/07, Thăng Long Compact (họng), Thăng Long (răng) | |
| - Đơn vị: cây/bao, chu kỳ: ngày/tuần/tháng | |
| - Thông điệp CEO-friendly, kênh: GT/MT/Online | |
| """ | |
| from datetime import datetime, timedelta | |
| import random | |
| import numpy as np | |
| import plotly.graph_objects as go | |
| def _z_from_service_level(p: float) -> float: | |
| """Ước lượng Z cho mức dịch vụ (để tính tồn an toàn/ROP).""" | |
| if p >= 0.99: | |
| return 2.33 | |
| if p >= 0.975: | |
| return 1.96 | |
| if p >= 0.95: | |
| return 1.65 | |
| if p >= 0.90: | |
| return 1.28 | |
| return 1.0 | |
| def _baseline_per_point(don_vi_co_ban: float, | |
| don_vi_tinh: str, | |
| che_do_xem: str) -> float: | |
| """ | |
| Chuẩn hóa baseline theo đơn vị thời gian hiển thị. | |
| Ví dụ: don_vi_co_ban=180 (cây/tháng) -> mỗi điểm ngày ≈ 180/30.44. | |
| """ | |
| don_vi_tinh = (don_vi_tinh or "").lower().strip() | |
| if "/thang" in don_vi_tinh or "/tháng" in don_vi_tinh: | |
| per_month = don_vi_co_ban | |
| if che_do_xem == "năm": | |
| return per_month # mỗi điểm là 1 tháng | |
| if che_do_xem == "tháng": | |
| return per_month / 4.33 # mỗi điểm là 1 tuần | |
| return per_month / 30.44 # ngày | |
| if "/tuan" in don_vi_tinh or "/tuần" in don_vi_tinh: | |
| per_week = don_vi_co_ban | |
| if che_do_xem == "năm": | |
| return per_week * 4.33 # gom thành tháng | |
| if che_do_xem == "tháng": | |
| return per_week | |
| return per_week / 7.0 | |
| if "/ngay" in don_vi_tinh or "/ngày" in don_vi_tinh: | |
| per_day = don_vi_co_ban | |
| if che_do_xem == "năm": | |
| return per_day * 30.44 | |
| if che_do_xem == "tháng": | |
| return per_day * 7.0 | |
| return per_day | |
| # Mặc định coi là theo tháng | |
| if che_do_xem == "năm": | |
| return don_vi_co_ban | |
| if che_do_xem == "tháng": | |
| return don_vi_co_ban / 4.33 | |
| return don_vi_co_ban / 30.44 | |
| def tao_bieu_do_mac_dinh(ds_sku=None): | |
| """ | |
| Tạo biểu đồ mặc định khi chưa dự đoán, hiển thị đúng domain SKU. | |
| ds_sku: optional list tên SKU để gợi ý chọn. | |
| """ | |
| fig = go.Figure() | |
| sku_text = "" | |
| if ds_sku: | |
| sku_text = " | ".join(ds_sku) | |
| else: | |
| try: | |
| # Lấy từ TRUONG_HOP_DEMO nếu có trong scope | |
| sku_text = " | ".join(list(TRUONG_HOP_DEMO.keys())[:3]) | |
| except Exception: | |
| sku_text = "Thủ Đô NH214/07 | TL Compact (họng) | TL (răng)" | |
| fig.add_annotation( | |
| text=( | |
| "📊 Chọn SKU và nhấn <b>“Tạo Dự Báo AI”</b><br>" | |
| "SKU mẫu: " + sku_text | |
| ), | |
| xref="paper", | |
| yref="paper", | |
| x=0.5, | |
| y=0.55, | |
| xanchor="center", | |
| yanchor="middle", | |
| showarrow=False, | |
| font=dict(size=18, color="#34495E", family="Arial"), | |
| bgcolor="rgba(255,255,255,0.9)", | |
| bordercolor="#BDC3C7", | |
| borderwidth=2, | |
| borderpad=20, | |
| ) | |
| fig.add_annotation( | |
| text="Đơn vị hiển thị: cây (1 cây ≈ 10 bao)", | |
| xref="paper", | |
| yref="paper", | |
| x=0.5, | |
| y=0.40, | |
| xanchor="center", | |
| yanchor="middle", | |
| showarrow=False, | |
| font=dict(size=13, color="#566573", family="Arial"), | |
| bgcolor="rgba(255,255,255,0.6)", | |
| bordercolor="#D5D8DC", | |
| borderwidth=1, | |
| borderpad=8, | |
| ) | |
| fig.update_layout( | |
| title="🎯 Bảng Điều Hành Dự Báo | Thủ Đô & Thăng Long", | |
| template="plotly_white", | |
| height=450, | |
| showlegend=False, | |
| xaxis=dict(showgrid=False, showticklabels=False, title="", automargin=True), | |
| yaxis=dict(showgrid=False, showticklabels=False, title="", automargin=True), | |
| plot_bgcolor="rgba(248,249,250,0.9)", | |
| paper_bgcolor="rgba(255,255,255,0.95)", | |
| font=dict(family="Inter, Arial, sans-serif", size=14), | |
| margin=dict(l=40, r=40, t=80, b=40), | |
| ) | |
| return fig | |
| def tao_tom_tat_mac_dinh(): | |
| """Tạo tóm tắt mặc định theo domain SKU.""" | |
| return """ | |
| <div class="prediction-box"> | |
| <h3 style="margin-top:0;color:white;">📋 Tóm Tắt Sẽ Hiển Thị Sau Khi Dự Báo</h3> | |
| <div style="text-align:center;padding:20px;"> | |
| <p style="font-size:18px;margin:10px 0;color:white;"> | |
| 🎯 Chọn SKU (Thủ Đô / Thăng Long) và nhấn "Tạo Dự Báo AI" | |
| </p> | |
| <p style="font-size:16px;margin:10px 0;color:#E8F6F3;"> | |
| Hệ thống sẽ tính xu hướng, tồn an toàn, ROP và cảnh báo kênh GT/MT | |
| </p> | |
| </div> | |
| </div> | |
| """ | |
| def tao_du_lieu_demo(thong_tin_case, chu_ky="3 tháng", che_do_xem="ngày"): | |
| """ | |
| Tạo dữ liệu demo đơn giản dựa trên case và chu kỳ dự báo. | |
| Domain: đơn vị 'cây'; tự động chuẩn hóa baseline theo thời gian hiển thị. | |
| """ | |
| don_vi_co_ban = float(thong_tin_case.get("don_vi_co_ban", 100)) | |
| xu_huong = (thong_tin_case.get("xu_huong") or "").lower().strip() | |
| don_vi_tinh = thong_tin_case.get("don_vi_tinh", "cay/thang") | |
| # Seed ổn định | |
| seed_value = hash( | |
| (thong_tin_case.get("ma_sku", ""), chu_ky, che_do_xem) | |
| ) & 0xFFFFFFFF | |
| rng = random.Random(seed_value) | |
| # Số điểm dữ liệu | |
| if chu_ky == "1 năm": | |
| so_diem = 12 if che_do_xem == "năm" else (52 if che_do_xem == "tháng" else 365) | |
| elif chu_ky == "6 tháng": | |
| so_diem = 6 if che_do_xem == "năm" else (26 if che_do_xem == "tháng" else 180) | |
| else: # "3 tháng" | |
| so_diem = 3 if che_do_xem == "năm" else (13 if che_do_xem == "tháng" else 90) | |
| # Nhãn thời gian | |
| if che_do_xem == "năm": | |
| nhan_thoi_gian = [f"Tháng {i + 1}" for i in range(so_diem)] | |
| elif che_do_xem == "tháng": | |
| nhan_thoi_gian = [f"Tuần {i + 1}" for i in range(so_diem)] | |
| else: # ngày | |
| ngay_bat_dau = datetime.now() | |
| nhan_thoi_gian = [ | |
| (ngay_bat_dau + timedelta(days=i)).strftime("%d/%m") for i in range(so_diem) | |
| ] | |
| # Baseline theo mỗi điểm | |
| base_per_point = _baseline_per_point(don_vi_co_ban, don_vi_tinh, che_do_xem) | |
| # Dữ liệu theo xu hướng domain | |
| don_vi_ban = [] | |
| for i in range(so_diem): | |
| if xu_huong in {"tang_dan", "tăng_dần"}: | |
| he_so = 1.0 + i * 0.20 / max(1, so_diem // 12 or 1) | |
| elif xu_huong in {"tang_nhe", "tăng_nhẹ"}: | |
| he_so = 1.0 + i * 0.08 / max(1, so_diem // 12 or 1) | |
| elif xu_huong in {"bien_dong", "biến_động"}: | |
| he_so = 1.0 + np.sin(i * 0.5) * 0.3 | |
| elif xu_huong in {"on_dinh", "ổn_định"}: | |
| he_so = 1.0 + rng.uniform(-0.1, 0.1) | |
| else: | |
| he_so = 1.0 | |
| noise = rng.randint(-3, 3) if che_do_xem == "ngày" else rng.randint(-7, 7) | |
| gia_tri = max(int(base_per_point * he_so + noise), 1) | |
| don_vi_ban.append(gia_tri) | |
| return nhan_thoi_gian, don_vi_ban | |
| def tao_bieu_do_du_doan( | |
| ten_case, | |
| chu_ky="3 tháng", | |
| che_do_xem="ngày", | |
| df_user=None, | |
| cot_ngay=None, | |
| cot_gia_tri=None, | |
| ): | |
| """ | |
| Tạo biểu đồ dự đoán cho SKU domain hiện tại. | |
| Luôn dùng dữ liệu demo (không phân tích CSV thật). | |
| """ | |
| thong_tin_case = TRUONG_HOP_DEMO[ten_case] | |
| nhan_thoi_gian, don_vi_ban = tao_du_lieu_demo( | |
| thong_tin_case, chu_ky, che_do_xem | |
| ) | |
| fig = go.Figure() | |
| # Biểu đồ chính | |
| fig.add_trace( | |
| go.Scatter( | |
| x=nhan_thoi_gian, | |
| y=don_vi_ban, | |
| mode="lines+markers", | |
| name="📦 Sản Lượng (cây)", | |
| line=dict(color="#1A5F91", width=4), | |
| marker=dict(size=10, color="#1A5F91", symbol="circle"), | |
| hovertemplate="<b>%{x}</b><br><b>Sản lượng:</b> %{y:,} cây<extra></extra>", | |
| ) | |
| ) | |
| # Vùng tin cậy ±15% | |
| upper_bound = [val * 1.15 for val in don_vi_ban] | |
| lower_bound = [val * 0.85 for val in don_vi_ban] | |
| fig.add_trace( | |
| go.Scatter( | |
| x=nhan_thoi_gian + nhan_thoi_gian[::-1], | |
| y=upper_bound + lower_bound[::-1], | |
| fill="toself", | |
| fillcolor="rgba(26, 95, 145, 0.2)", | |
| line=dict(color="rgba(255,255,255,0)"), | |
| name="📊 Vùng Tin Cậy", | |
| hoverinfo="skip", | |
| ) | |
| ) | |
| # Đường xu hướng tuyến tính | |
| x_so = list(range(len(nhan_thoi_gian))) | |
| if len(x_so) >= 2: | |
| z = np.polyfit(x_so, don_vi_ban, 1) | |
| duong_xu_huong = np.poly1d(z)(x_so) | |
| fig.add_trace( | |
| go.Scatter( | |
| x=nhan_thoi_gian, | |
| y=duong_xu_huong, | |
| mode="lines", | |
| name="📈 Xu Hướng", | |
| line=dict(color="#8B8B8B", width=3, dash="dash"), | |
| hovertemplate="<b>Xu hướng:</b> %{y:.0f} cây<extra></extra>", | |
| ) | |
| ) | |
| # Đỉnh doanh số | |
| if don_vi_ban: | |
| chi_so = int(np.argmax(don_vi_ban)) | |
| fig.add_trace( | |
| go.Scatter( | |
| x=[nhan_thoi_gian[chi_so]], | |
| y=[don_vi_ban[chi_so]], | |
| mode="markers", | |
| name="🔥 Đỉnh Doanh Số", | |
| marker=dict(size=18, color="#F39C12", symbol="star"), | |
| hovertemplate="<b>Đỉnh:</b> %{x}<br><b>Sản lượng:</b> %{y:,} cây<extra></extra>", | |
| ) | |
| ) | |
| # Tiêu đề theo chế độ xem | |
| tieu_de_che_do = {"ngày": "Theo Ngày", "tháng": "Theo Tuần", "năm": "Theo Tháng"} | |
| fig.update_layout( | |
| title=f"📊 Dự Báo Sản Lượng Tút - {tieu_de_che_do[che_do_xem]}: {ten_case} ({chu_ky})", | |
| xaxis_title="⏰ Thời Gian", | |
| yaxis_title="📦 Số Tút tiêu thụ", | |
| template="plotly_white", | |
| height=650, | |
| showlegend=True, | |
| hovermode="x unified", | |
| xaxis=dict( | |
| tickangle=45 if che_do_xem == "ngày" else 0, | |
| fixedrange=True, | |
| tickmode="auto", | |
| nticks=10, | |
| automargin=True, | |
| ), | |
| yaxis=dict(fixedrange=True, automargin=True), | |
| font=dict(size=12, family="Inter, Arial, sans-serif"), | |
| plot_bgcolor="rgba(248,249,250,0.9)", | |
| paper_bgcolor="rgba(255,255,255,0.95)", | |
| margin=dict(l=80, r=60, t=120, b=100), | |
| uirevision=f"{ten_case}|{chu_ky}|{che_do_xem}|{len(nhan_thoi_gian)}", | |
| ) | |
| # Ẩn toolbar/config | |
| # Lưu ý: biến `config` sẽ được truyền cho component Plotly khi render (tùy framework). | |
| fig._config = { | |
| "displayModeBar": False, | |
| "staticPlot": True, | |
| "displaylogo": False, | |
| "responsive": True, | |
| "editable": False, | |
| "showTips": False, | |
| "watermark": False, | |
| } | |
| return fig | |
| def tao_thong_tin_tom_tat( | |
| ten_case, | |
| chu_ky="3 tháng", | |
| che_do_xem="ngày", | |
| ): | |
| """ | |
| Tạo tóm tắt điều hành cho SKU domain hiện tại. | |
| Tính thêm tồn an toàn và ROP nếu có 'muc_dich_vu' và 'lead_time_ngay'. | |
| """ | |
| thong_tin_case = TRUONG_HOP_DEMO[ten_case] | |
| nhan_thoi_gian, don_vi_ban = tao_du_lieu_demo( | |
| thong_tin_case, chu_ky, che_do_xem | |
| ) | |
| tong_don_vi = int(sum(don_vi_ban)) | |
| trung_binh = float(tong_don_vi) / max(1, len(don_vi_ban)) | |
| dinh = int(max(don_vi_ban)) if don_vi_ban else 0 | |
| day_nhat = int(min(don_vi_ban)) if don_vi_ban else 0 | |
| ty_le_tang_truong = ( | |
| ((don_vi_ban[-1] - don_vi_ban[0]) / max(1, don_vi_ban[0])) * 100 | |
| if len(don_vi_ban) >= 2 | |
| else 0.0 | |
| ) | |
| # Ước lượng tồn an toàn/ROP theo chu kỳ hiển thị hiện tại | |
| muc_dich_vu = float(thong_tin_case.get("muc_dich_vu", 0.95)) | |
| lead_time_ngay = int(thong_tin_case.get("lead_time_ngay", 14)) | |
| z = _z_from_service_level(muc_dich_vu) | |
| # Ước lượng sigma theo chu kỳ hiện tại | |
| sigma = float(np.std(don_vi_ban)) if len(don_vi_ban) > 1 else 0.0 | |
| lt_weeks = max(1.0, lead_time_ngay / 7.0) | |
| ton_an_toan = z * sigma * np.sqrt(lt_weeks) | |
| rop = trung_binh * (lead_time_ngay / 7.0) + ton_an_toan | |
| # Kênh chủ lực | |
| ty_le_kenh = thong_tin_case.get("ty_le_kenh") | |
| if isinstance(ty_le_kenh, dict) and ty_le_kenh: | |
| kenh_chu_luc = max(ty_le_kenh.items(), key=lambda x: x[1])[0] | |
| else: | |
| ds_kenh = thong_tin_case.get("kenh_ban", []) | |
| kenh_chu_luc = ds_kenh[0] if ds_kenh else "GT" | |
| don_vi_thoi_gian = {"ngày": "ngày", "tháng": "tuần", "năm": "tháng"}[che_do_xem] | |
| san_pham = thong_tin_case.get("san_pham", "") | |
| canh_bao = thong_tin_case.get("canh_bao") | |
| canh_bao_txt = f" | Cảnh báo: {canh_bao}" if canh_bao else "" | |
| tom_tat_html = f""" | |
| <div class="prediction-box"> | |
| <h3 style="margin-top:0;color:white;font-size:1.8em;font-weight:700;text-align:center;margin-bottom:24px;"> | |
| 📋 Báo Cáo Điều Hành | {san_pham}{canh_bao_txt} | |
| </h3> | |
| <div style="display:grid;grid-template-columns:repeat(2,1fr);gap:20px;margin:16px 0;"> | |
| <div style="background:rgba(255,255,255,0.25);padding:16px;border-radius:12px;text-align:center;border:1px solid rgba(255,255,255,0.3);"> | |
| <h4 style="margin:0 0 8px 0;color:white;font-size:1.1em;font-weight:600;">📦 Tổng Sản Lượng</h4> | |
| <p style="font-size:28px;margin:8px 0;font-weight:800;color:white;">{tong_don_vi:,} cây</p> | |
| </div> | |
| <div style="background:rgba(255,255,255,0.25);padding:16px;border-radius:12px;text-align:center;border:1px solid rgba(255,255,255,0.3);"> | |
| <h4 style="margin:0 0 8px 0;color:white;font-size:1.1em;font-weight:600;">📈 TB/{don_vi_thoi_gian.title()}</h4> | |
| <p style="font-size:28px;margin:8px 0;font-weight:800;color:white;">{trung_binh:,.0f} cây</p> | |
| </div> | |
| <div style="background:rgba(255,255,255,0.25);padding:16px;border-radius:12px;text-align:center;border:1px solid rgba(255,255,255,0.3);"> | |
| <h4 style="margin:0 0 8px 0;color:white;font-size:1.1em;font-weight:600;">🔥 Đỉnh</h4> | |
| <p style="font-size:28px;margin:8px 0;font-weight:800;color:white;">{dinh:,} cây</p> | |
| </div> | |
| <div style="background:rgba(255,255,255,0.25);padding:16px;border-radius:12px;text-align:center;border:1px solid rgba(255,255,255,0.3);"> | |
| <h4 style="margin:0 0 8px 0;color:white;font-size:1.1em;font-weight:600;">📊 Tăng Trưởng</h4> | |
| <p style="font-size:28px;margin:8px 0;font-weight:800;color:{'#4ADE80' if ty_le_tang_truong > 0 else '#F87171'};"> | |
| {ty_le_tang_truong:+.1f}% | |
| </p> | |
| </div> | |
| </div> | |
| <div style="display:grid;grid-template-columns:repeat(3,1fr);gap:16px;margin-top:8px;"> | |
| <div style="background:rgba(255,255,255,0.18);padding:14px;border-radius:10px;border:1px solid rgba(255,255,255,0.25);text-align:center;"> | |
| <div style="color:white;font-weight:600;">🏬 Kênh Chủ Lực</div> | |
| <div style="color:white;margin-top:6px;">{kenh_chu_luc}</div> | |
| </div> | |
| <div style="background:rgba(255,255,255,0.18);padding:14px;border-radius:10px;border:1px solid rgba(255,255,255,0.25);text-align:center;"> | |
| <div style="color:white;font-weight:600;">🛡️ Tồn An Toàn (ước tính)</div> | |
| <div style="color:white;margin-top:6px;">{ton_an_toan:,.1f} cây/tuần</div> | |
| </div> | |
| <div style="background:rgba(255,255,255,0.18);padding:14px;border-radius:10px;border:1px solid rgba(255,255,255,0.25);text-align:center;"> | |
| <div style="color:white;font-weight:600;">🔔 ROP (điểm đặt hàng lại)</div> | |
| <div style="color:white;margin-top:6px;">{rop:,.1f} cây</div> | |
| </div> | |
| </div> | |
| <div style="margin-top:18px;text-align:center;padding:12px;background:rgba(255,255,255,0.15);border-radius:10px;border:1px solid rgba(255,255,255,0.2);"> | |
| <p style="margin:0;font-size:1.05em;color:white;font-weight:600;"> | |
| 📅 Chu Kỳ Dự Báo: <strong>{chu_ky}</strong> | Mức dịch vụ mục tiêu: <strong>{muc_dich_vu:.0%}</strong> | LT: <strong>{lead_time_ngay} ngày</strong> | |
| </p> | |
| </div> | |
| </div> | |
| """ | |
| return tom_tat_html | |
| def tao_bao_cao_giam_doc(ten_case): | |
| """ | |
| Tạo báo cáo chi tiết dành cho giám đốc (Markdown), | |
| giữ nội dung domain từ TRUONG_HOP_DEMO. | |
| """ | |
| thong_tin_case = TRUONG_HOP_DEMO[ten_case] | |
| thong_tin_gd = thong_tin_case["thong_tin_giam_doc"] | |
| bao_cao_markdown = f""" | |
| # 🎯 CHIẾN LƯỢC ĐỀ XUẤT | |
| ## 📈 Tóm Tắt Điều Hành | |
| {thong_tin_gd['tom_tat']} | |
| ## 📊 Phân Tích Chi Tiết | |
| {thong_tin_gd['chi_tiet']} | |
| ## 🚀 Khuyến Nghị Hành Động | |
| {thong_tin_gd['hanh_dong']} | |
| --- | |
| *🤖 FoxAI Business Intelligence | ⚡ Cập nhật realtime | Đơn vị: cây (1 cây ≈ 10 bao)* | |
| """ | |
| return bao_cao_markdown | |
| def tao_loading_effect(): | |
| """Tạo hiệu ứng loading để có cảm giác như production""" | |
| loading_html = """ | |
| <div class="loading-container"> | |
| <div class="loading-spinner"></div> | |
| <h3 style="color: white; margin: 0;">🤖 Đang Xử Lý Dự Báo AI...</h3> | |
| <div class="progress-bar"> | |
| <div class="progress-fill"></div> | |
| </div> | |
| <p style="color: white; margin: 10px 0; opacity: 0.9;"> | |
| ⚡ Phân tích dữ liệu • 🧠 Machine Learning • 📊 Tạo insights • 🎯 Chiến lược AI | |
| </p> | |
| </div> | |
| """ | |
| return loading_html | |
| def tao_bao_cao_streaming(ten_case): | |
| """Tạo báo cáo với hiệu ứng streaming như AI đang viết - sử dụng module streaming mới""" | |
| # Trả về string thay vì generator để tương thích với Gradio | |
| return tao_bao_cao_noi_dung_streaming(ten_case) | |
| def tao_streaming_text(ten_case): | |
| """Generator để tạo hiệu ứng streaming - alias cho hàm mới""" | |
| # Trả về string thay vì generator để tương thích với Gradio | |
| return tao_bao_cao_noi_dung_streaming(ten_case) | |
| def xu_ly_du_doan_voi_loading(lua_chon_case, chu_ky, che_do_xem): | |
| """Xử lý dự đoán với hiệu ứng loading realistic và streaming - sử dụng module mới""" | |
| # Sử dụng hàm streaming mới với streaming=False để tương thích với Gradio | |
| return xu_ly_du_doan_voi_loading_moi(lua_chon_case, chu_ky, che_do_xem, streaming=False) | |
| def cap_nhat_thong_tin_case(lua_chon_case): | |
| """Cập nhật thông tin case khi chọn - sử dụng hàm mới""" | |
| return cap_nhat_thong_tin_case_moi(lua_chon_case) | |
| def tao_csv_gia_lap(ten_case, chu_ky, che_do_xem): | |
| """Tạo CSV giả lập để demo - sử dụng hàm mới""" | |
| return tao_csv_gia_lap_moi(ten_case, chu_ky, che_do_xem) | |
| def bat_dau_loading(): | |
| """Khởi động: show panel loading ngay để auto-scroll rồi mới stream - sử dụng hàm mới""" | |
| return bat_dau_loading_moi() | |
| def tinh_toan_ket_qua( | |
| lua_chon_case, chu_ky, che_do_xem | |
| ): | |
| """Tính toán kết quả thật sự - sử dụng hàm streaming mới""" | |
| return tinh_toan_ket_qua_moi(lua_chon_case, chu_ky, che_do_xem, streaming=False) | |
| # Tạo ứng dụng Gradio | |
| def tao_ung_dung(): | |
| """Tạo ứng dụng Gradio hoàn chỉnh bằng tiếng Việt""" | |
| with gr.Blocks( | |
| title="🚀 Hệ Thống Dự Báo Bán Hàng AI - FoxAI", | |
| theme=gr.themes.Soft(), | |
| css=custom_css | |
| ) as demo: | |
| gr.HTML(f""" | |
| <div class="header-container"> | |
| <!-- Floating squares background --> | |
| <div class="floating-square"></div> | |
| <div class="floating-square"></div> | |
| <div class="floating-square"></div> | |
| <div class="floating-square"></div> | |
| <div class="floating-square"></div> | |
| <div class="floating-square"></div> | |
| <!-- Geometric shapes --> | |
| <div class="geometric-shapes"> | |
| <div class="shape triangle"></div> | |
| <div class="shape circle"></div> | |
| <div class="shape hexagon"></div> | |
| <div class="shape diamond"></div> | |
| </div> | |
| <!-- Grid overlay --> | |
| <div class="grid-overlay"></div> | |
| <!-- Wave container --> | |
| <div class="wave-container"> | |
| <div class="wave"></div> | |
| <div class="wave"></div> | |
| <div class="wave"></div> | |
| </div> | |
| <!-- Shooting stars --> | |
| <div class="shooting-star"></div> | |
| <div class="shooting-star"></div> | |
| <div class="shooting-star"></div> | |
| <!-- Particle overlay --> | |
| <div class="particle-overlay"></div> | |
| <style> | |
| .header-container {{ | |
| display: flex; | |
| flex-direction: column; | |
| align-items: center; | |
| padding: 24px; | |
| position: relative; | |
| overflow: hidden; | |
| }} | |
| .foxai-logo {{ | |
| height: 64px; | |
| width: auto; | |
| filter: drop-shadow(0 6px 14px rgba(56,189,248,.45)); | |
| transition: transform .25s ease; | |
| }} | |
| .foxai-logo:hover {{ | |
| transform: scale(1.05); | |
| }} | |
| .header-title {{ | |
| margin-top: 12px; | |
| font-size: clamp(22px, 3vw, 34px); | |
| font-weight: 800; | |
| letter-spacing: .3px; | |
| background: linear-gradient(90deg, #a5b4fc, #60a5fa, #34d399, #60a5fa); | |
| background-size: 200% 100%; | |
| -webkit-background-clip: text; | |
| background-clip: text; | |
| color: transparent; | |
| animation: title-shift 6s ease-in-out infinite; | |
| }} | |
| @keyframes title-shift {{ | |
| 0% {{ background-position: 0% 50%; }} | |
| 50% {{ background-position: 100% 50%; }} | |
| 100% {{ background-position: 0% 50%; }} | |
| }} | |
| .header-container p {{ | |
| font-size: 1.05em; | |
| color: rgba(255,255,255,0.85); | |
| margin: 12px auto 10px; | |
| font-weight: 500; | |
| line-height: 1.55; | |
| max-width: 640px; | |
| text-align: center; | |
| }} | |
| .header-container p strong {{ | |
| color: #fff; | |
| }} | |
| .system-status {{ | |
| display: inline-flex; | |
| align-items: center; | |
| gap: 8px; | |
| padding: 6px 14px; | |
| border-radius: 999px; | |
| border: 1px solid rgba(255,255,255,.1); | |
| background: rgba(255,255,255,.06); | |
| font-size: 0.9em; | |
| color: #fff; | |
| margin-top: 14px; | |
| }} | |
| .system-status::before {{ | |
| content: ""; | |
| width: 10px; height: 10px; | |
| border-radius: 50%; | |
| background: #22c55e; | |
| box-shadow: 0 0 0 0 rgba(34,197,94,.6); | |
| animation: pulse 2s infinite; | |
| }} | |
| @keyframes pulse {{ | |
| 0% {{ box-shadow: 0 0 0 0 rgba(34,197,94,.6); }} | |
| 70% {{ box-shadow: 0 0 0 10px rgba(34,197,94,0); }} | |
| 100% {{ box-shadow: 0 0 0 0 rgba(34,197,94,0); }} | |
| }} | |
| </style> | |
| <div style="text-align: center; margin-bottom: 12px;"> | |
| <img src="{FOXAI_LOGO_URL}" alt="FoxAI Logo" class="foxai-logo"/> | |
| </div> | |
| <div class="header-title">HỆ THỐNG DỰ BÁO BÁN HÀNG AI</div> | |
| <p> | |
| Phân tích thông minh với công nghệ <strong>FoxAI</strong> | |
| </p> | |
| <div class="system-status"> | |
| Hoạt động • Tối ưu mô hình • FoxAI Native | |
| </div> | |
| </div> | |
| """) | |
| gr.HTML(""" | |
| <div class='info-card' style=" | |
| padding: 35px; | |
| border-radius: 20px; | |
| border: 1px solid rgba(255,255,255,0.15); | |
| background: rgba(30,41,59,0.75); /* nền tối mờ, đồng bộ footer */ | |
| backdrop-filter: blur(10px); | |
| box-shadow: 0 8px 25px rgba(0,0,0,0.25); | |
| text-align: center; | |
| "> | |
| <h3 style=" | |
| margin-bottom: 20px; | |
| font-size: 2em; | |
| font-weight: 800; | |
| color: #fff; /* chữ trắng rõ ràng */ | |
| text-shadow: 0 2px 6px rgba(0,0,0,0.25); | |
| "> | |
| 🎯 Hướng Dẫn Sử Dụng | |
| </h3> | |
| <p style=" | |
| font-size: 1.15em; | |
| margin-bottom: 30px; | |
| color: #f1f5f9; | |
| font-weight: 500; | |
| "> | |
| Chọn sản phẩm, chu kỳ dự báo và nhận phân tích AI chi tiết | |
| </p> | |
| <div style=" | |
| display: grid; | |
| grid-template-columns: repeat(auto-fit, minmax(280px, 1fr)); | |
| gap: 25px; | |
| margin: 25px 0; | |
| "> | |
| <!-- Card 1 --> | |
| <div style=" | |
| padding: 24px; | |
| border-radius: 16px; | |
| border: 1px solid rgba(255,255,255,0.1); | |
| background: rgba(17,25,40,0.8); /* slate đậm hơn chút */ | |
| box-shadow: 0 6px 16px rgba(0,0,0,0.25); | |
| "> | |
| <h4 style=" | |
| margin-bottom: 15px; | |
| font-size: 1.2em; | |
| font-weight: 700; | |
| color: #fff; | |
| ">🛍️ Danh Mục Sản Phẩm</h4> | |
| <div style=" | |
| color: #e2e8f0; | |
| line-height: 1.8; | |
| font-weight: 500; | |
| font-size: 1.05em; | |
| "> | |
| Thủ đô bao cứng<br/> | |
| Thăng Long bao cứng Compact (cảnh báo họng)<br/> | |
| Thăng Long cảnh báo răng | |
| </div> | |
| </div> | |
| <!-- Card 2 --> | |
| <div style=" | |
| padding: 24px; | |
| border-radius: 16px; | |
| border: 1px solid rgba(255,255,255,0.1); | |
| background: rgba(17,25,40,0.8); | |
| box-shadow: 0 6px 16px rgba(0,0,0,0.25); | |
| "> | |
| <h4 style=" | |
| margin-bottom: 15px; | |
| font-size: 1.2em; | |
| font-weight: 700; | |
| color: #fff; | |
| ">📊 Chu Kỳ Dự Báo</h4> | |
| <div style=" | |
| color: #e2e8f0; | |
| line-height: 1.8; | |
| font-weight: 500; | |
| font-size: 1.05em; | |
| "> | |
| 📅 <strong>3 Tháng:</strong> Ngắn hạn<br/> | |
| 📆 <strong>6 Tháng:</strong> Trung hạn<br/> | |
| 🗓️ <strong>1 Năm:</strong> Dài hạn | |
| </div> | |
| </div> | |
| </div> | |
| </div> | |
| """) | |
| # Sử dụng Tabs để tổ chức giao diện | |
| with gr.Tabs(): | |
| # Tab 1: Dự báo chính | |
| with gr.Tab("📊 Dự Báo Bán Hàng"): | |
| with gr.Row(): | |
| with gr.Column(scale=1): | |
| gr.HTML("<div class='feature-box floating' style='text-align: center;'><h3>⚙️ Cài Đặt</h3></div>") | |
| lua_chon_dropdown = gr.Dropdown( | |
| choices=list(TRUONG_HOP_DEMO.keys()), | |
| value=list(TRUONG_HOP_DEMO.keys())[0], | |
| label="🛍️ Chọn Sản Phẩm", | |
| info="Mỗi sản phẩm có insight riêng", | |
| elem_classes=["tech-border", "white-info-dropdown"] | |
| ) | |
| chu_ky_dropdown = gr.Dropdown( | |
| choices=[("📅 3 Tháng", "3 tháng"), ("📆 6 Tháng", "6 tháng"), ("🗓️ 1 Năm", "1 năm")], | |
| value="3 tháng", | |
| label="📊 Chu Kỳ Dự Báo", | |
| info="Chọn khoảng thời gian phù hợp", | |
| elem_classes=["tech-border", "white-info-dropdown"] | |
| ) | |
| nut_du_bao = gr.Button( | |
| "🚀 Tạo Dự Báo AI", | |
| variant="primary", | |
| size="lg", | |
| elem_classes=["btn-modern", "neon-border"] | |
| ) | |
| # Hiển thị thông tin CSV giả lập | |
| gr.HTML(""" | |
| <div class='info-card'> | |
| <p><strong>Trạng thái kết nối cơ sở dữ liệu:</strong> Ổn định</p> | |
| <p><strong>Định dạng:</strong> Time series data với cột thời gian và doanh thu</p> | |
| <p><strong>Cập nhật:</strong> Tự động sync theo mặt hàng được chọn</p> | |
| </div> | |
| """) | |
| # Hiển thị thông tin case | |
| hien_thi_thong_tin = gr.HTML() | |
| with gr.Row(): | |
| with gr.Column(): | |
| gr.HTML("<div class='feature-box floating' style='text-align: center;'><h3>📈 Kết Quả Dự Báo</h3></div>") | |
| # Chế độ hiển thị được đặt gần biểu đồ | |
| che_do_xem = gr.Radio( | |
| choices=[("📅 Theo Ngày", "ngày"), ("📆 Theo Tuần", "tháng"), ("🗓️ Theo Tháng", "năm")], | |
| value="ngày", | |
| label="📊 Chế Độ Hiển Thị", | |
| info="Thay đổi cách hiển thị biểu đồ", | |
| elem_classes=["tech-border"] | |
| ) | |
| bieu_do_du_bao = gr.Plot( | |
| label="Biểu Đồ Dự Báo Bán Hàng", | |
| elem_classes=["plot-container", "glass-card", "neon-border"], | |
| elem_id="chart-target" # Thêm ID để có thể cuộn tới | |
| ) | |
| # Panel loading | |
| loading_panel = gr.HTML(value="", elem_id="loading-panel") | |
| with gr.Row(): | |
| with gr.Column(scale=1): | |
| gr.HTML("<div class='feature-box floating' style='text-align: center;'><h3>📊 Thống Kê</h3></div>") | |
| hien_thi_tom_tat = gr.HTML() | |
| with gr.Column(scale=1): | |
| gr.HTML("<div class='feature-box floating' style='text-align: center;'><h3>🎯 Chiến Lược AI</h3></div>") | |
| # Button để tạo streaming effect | |
| nut_streaming = gr.Button( | |
| "🤖 Tạo Chiến Lược AI (Streaming)", | |
| variant="secondary", | |
| size="sm", | |
| elem_classes=["btn-modern"], | |
| visible=False | |
| ) | |
| hien_thi_bao_cao = gr.Markdown() | |
| # Tab 2: Các tính năng đang phát triển | |
| with gr.Tab("🔬 Tính Năng Đang Phát Triển"): | |
| gr.HTML(""" | |
| <div class='feature-box floating' style='text-align: center; margin-bottom: 30px;'> | |
| <h3>🔬 Các Tính Năng Đang Phát Triển</h3> | |
| <p style='color: white; font-size: 1.1em; margin-top: 15px;'> | |
| Những công nghệ tiên tiến đang được nghiên cứu và phát triển | |
| </p> | |
| </div> | |
| """) | |
| with gr.Row(): | |
| with gr.Column(): | |
| # Dự báo đa mô hình | |
| gr.HTML(""" | |
| <div class='modern-card holographic'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.6em;'> | |
| 🧠 Dự báo đa mô hình (Ensemble Forecasting) | |
| </h3> | |
| <p style='color: white; font-size: 1.1em; line-height: 1.7; margin-bottom: 15px;'> | |
| Kết hợp nhiều kiến trúc AI (<strong>RNN, Transformer, Prophet, XGBoost...</strong>) | |
| để tăng độ chính xác và ổn định của dự báo. | |
| </p> | |
| <div style='background: rgba(255,255,255,0.1); padding: 15px; border-radius: 10px; margin-top: 15px;'> | |
| <strong style='color: #4ADE80;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047;'>Đang phát triển - 70% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| # Adaptive Learning | |
| gr.HTML(""" | |
| <div class='modern-card bg-blue-grad'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.6em;'> | |
| 🎯 Adaptive Learning theo thị trường | |
| </h3> | |
| <p style='color: white; font-size: 1.1em; line-height: 1.7; margin-bottom: 15px;'> | |
| 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). | |
| </p> | |
| <div style='background: rgba(255,255,255,0.1); padding: 15px; border-radius: 10px; margin-top: 15px;'> | |
| <strong style='color: #4ADE80;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047;'>Đang phát triển - 45% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| # Explainable AI | |
| gr.HTML(""" | |
| <div class='modern-card bg-gray-grad'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.6em;'> | |
| 🔍 Explainable AI (XAI) | |
| </h3> | |
| <p style='color: white; font-size: 1.1em; line-height: 1.7; margin-bottom: 15px;'> | |
| 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). | |
| </p> | |
| <div style='background: rgba(255,255,255,0.1); padding: 15px; border-radius: 10px; margin-top: 15px;'> | |
| <strong style='color: #4ADE80;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047;'>Đang phát triển - 60% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| with gr.Column(): | |
| # Real-time Forecast | |
| gr.HTML(""" | |
| <div class='modern-card tech-border'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.6em;'> | |
| ⚡ Real-time Forecast & Alerting | |
| </h3> | |
| <p style='color: white; font-size: 1.1em; line-height: 1.7; margin-bottom: 15px;'> | |
| 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. | |
| </p> | |
| <div style='background: rgba(255,255,255,0.1); padding: 15px; border-radius: 10px; margin-top: 15px;'> | |
| <strong style='color: #4ADE80;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047;'>Đang phát triển - 30% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| # Scenario Simulation | |
| gr.HTML(""" | |
| <div class='modern-card neon-border'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.6em;'> | |
| 🎲 Scenario Simulation (What-if Analysis) | |
| </h3> | |
| <p style='color: white; font-size: 1.1em; line-height: 1.7; margin-bottom: 15px;'> | |
| 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. | |
| </p> | |
| <div style='background: rgba(255,255,255,0.1); padding: 15px; border-radius: 10px; margin-top: 15px;'> | |
| <strong style='color: #4ADE80;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047;'>Đang phát triển - 25% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| # Forecast Granularity | |
| gr.HTML(""" | |
| <div class='modern-card floating'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.6em;'> | |
| 📊 Forecast Granularity | |
| </h3> | |
| <p style='color: white; font-size: 1.1em; line-height: 1.7; margin-bottom: 15px;'> | |
| 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. | |
| </p> | |
| <div style='background: rgba(255,255,255,0.1); padding: 15px; border-radius: 10px; margin-top: 15px;'> | |
| <strong style='color: #4ADE80;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047;'>Đang phát triển - 55% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| # Generative Insights - Full width | |
| gr.HTML(""" | |
| <div class='modern-card holographic' style='margin-top: 25px;'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.8em; text-align: center;'> | |
| 🤖 Generative Insights (AI Analyst) | |
| </h3> | |
| <p style='color: white; font-size: 1.2em; line-height: 1.8; margin-bottom: 20px; text-align: center;'> | |
| 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. | |
| </p> | |
| <div style='display: grid; grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); gap: 20px; margin-top: 25px;'> | |
| <div style='background: rgba(255,255,255,0.15); padding: 20px; border-radius: 12px; text-align: center;'> | |
| <h4 style='color: #4ADE80; margin-bottom: 10px;'>📝 Báo cáo tự động</h4> | |
| <p style='color: white; font-size: 0.95em;'>Tạo insights dạng văn bản từ dữ liệu</p> | |
| </div> | |
| <div style='background: rgba(255,255,255,0.15); padding: 20px; border-radius: 12px; text-align: center;'> | |
| <h4 style='color: #60A5FA; margin-bottom: 10px;'>🗣️ Voice Analytics</h4> | |
| <p style='color: white; font-size: 0.95em;'>Báo cáo bằng giọng nói tự nhiên</p> | |
| </div> | |
| <div style='background: rgba(255,255,255,0.15); padding: 20px; border-radius: 12px; text-align: center;'> | |
| <h4 style='color: #F472B6; margin-bottom: 10px;'>💬 Chat Interface</h4> | |
| <p style='color: white; font-size: 0.95em;'>Hỏi đáp trực tiếp với AI</p> | |
| </div> | |
| <div style='background: rgba(255,255,255,0.15); padding: 20px; border-radius: 12px; text-align: center;'> | |
| <h4 style='color: #34D399; margin-bottom: 10px;'>📧 Email Summary</h4> | |
| <p style='color: white; font-size: 0.95em;'>Gửi tóm tắt qua email tự động</p> | |
| </div> | |
| </div> | |
| <div style='background: rgba(255,255,255,0.1); padding: 20px; border-radius: 10px; margin-top: 25px; text-align: center;'> | |
| <strong style='color: #4ADE80; font-size: 1.1em;'>📈 Trạng thái:</strong> | |
| <span style='color: #FDE047; font-size: 1.1em;'>Đang phát triển - 40% hoàn thiện</span> | |
| </div> | |
| </div> | |
| """) | |
| MAP = "https://raw.githubusercontent.com/your-repo/heatmap-2025/main/heatmap.png" # Thay bằng URL thực tế của heatmap | |
| # Roadmap timeline | |
| # Roadmap timeline (update tháng 8/2025 → Q3 highlight) | |
| gr.HTML(f""" | |
| <div class='modern-card' style='margin-top: 30px; text-align: center;'> | |
| <h3 style='color: white; margin-bottom: 20px; font-size: 1.8em;'> | |
| 🔥 Heatmap Phân Tích 2025 | |
| </h3> | |
| <img src="{MAP}" | |
| alt="Heatmap" | |
| style="max-width: 100%; border-radius: 12px; box-shadow: 0 6px 18px rgba(0,0,0,0.25);" /> | |
| </div> | |
| """) | |
| # Xử lý sự kiện streaming cho chiến lược | |
| def xu_ly_streaming(lua_chon_case): | |
| """Xử lý streaming effect cho chiến lược đề xuất""" | |
| if lua_chon_case is None: | |
| return "# 🎯 CHIẾN LƯỢC ĐỀ XUẤT\n\n📊 **Chọn mặt hàng để tạo chiến lược**" | |
| try: | |
| for partial_text in tao_streaming_text(lua_chon_case): | |
| yield partial_text | |
| except Exception as e: | |
| yield f"# ❌ Lỗi\n\nKhông thể tạo chiến lược: {str(e)}" | |
| # Button streaming sẽ chỉ hiện khi đã có dữ liệu | |
| def hien_thi_nut_streaming(lua_chon_case): | |
| """Hiển thị button streaming khi đã chọn mặt hàng""" | |
| return gr.update(visible=lua_chon_case is not None) | |
| # Xử lý sự kiện cho Tab 1: Dự báo chính | |
| lua_chon_dropdown.change( | |
| fn=lambda x: [cap_nhat_thong_tin_case(x), hien_thi_nut_streaming(x)], | |
| inputs=[lua_chon_dropdown], | |
| outputs=[hien_thi_thong_tin, nut_streaming] | |
| ) | |
| # Event chain mới: scroll → loading 3–5s → render kết quả | |
| # 1) Kickoff: bật panel loading + auto scroll | |
| evt0 = nut_du_bao.click( | |
| fn=bat_dau_loading, | |
| inputs=[], | |
| outputs=[bieu_do_du_bao, hien_thi_tom_tat, hien_thi_bao_cao, loading_panel], | |
| show_progress=False, | |
| scroll_to_output=False, # Không scroll tự động, sẽ dùng JS | |
| ) | |
| # Scroll tới loading panel thay vì chart | |
| JS_SCROLL_TO_LOADING = r""" | |
| (...args) => { | |
| const el = document.getElementById('loading-panel'); | |
| if (el) { | |
| el.scrollIntoView({ behavior: 'smooth', block: 'center' }); | |
| el.classList.add('highlight'); | |
| setTimeout(() => el.classList.remove('highlight'), 1200); | |
| } | |
| return []; | |
| } | |
| """ | |
| try: | |
| evt0.then(fn=None, inputs=None, outputs=None, js=JS_SCROLL_TO_LOADING) | |
| except TypeError: | |
| evt0.then(fn=None, inputs=None, outputs=None, _js=JS_SCROLL_TO_LOADING) # type: ignore | |
| # 2) Stream 4 bước loading trong 3–5 giây (chỉ update loading_panel) | |
| evt1 = evt0.then( | |
| fn=hien_loading_tung_buoc, | |
| inputs=[], | |
| outputs=[loading_panel], | |
| show_progress=False, | |
| ) | |
| # 3) Tính toán thật, render kết quả + tắt loading | |
| evt2 = evt1.then( | |
| fn=tinh_toan_ket_qua, | |
| inputs=[lua_chon_dropdown, chu_ky_dropdown, che_do_xem], | |
| outputs=[bieu_do_du_bao, hien_thi_tom_tat, hien_thi_bao_cao, loading_panel], | |
| show_progress=False, | |
| ) | |
| # Thêm hiệu ứng fade-in cho LLM analysis sau khi render | |
| JS_FADE_IN_LLM = r""" | |
| (...args) => { | |
| setTimeout(() => { | |
| const markdownElements = document.querySelectorAll('.gradio-markdown'); | |
| markdownElements.forEach(el => { | |
| el.style.animation = 'fadeInUp 0.8s ease-out forwards'; | |
| }); | |
| }, 200); | |
| return []; | |
| } | |
| """ | |
| try: | |
| evt2.then(fn=None, inputs=None, outputs=None, js=JS_FADE_IN_LLM) | |
| except TypeError: | |
| evt2.then(fn=None, inputs=None, outputs=None, _js=JS_FADE_IN_LLM) # type: ignore | |
| # Streaming event cho chiến lược đề xuất | |
| nut_streaming.click( | |
| fn=xu_ly_streaming, | |
| inputs=[lua_chon_dropdown], | |
| outputs=[hien_thi_bao_cao], | |
| show_progress=False | |
| ) | |
| # Cập nhật biểu đồ khi thay đổi chế độ hiển thị - chỉ khi đã có dự đoán | |
| def cap_nhat_khi_doi_che_do(case, chu_ky, mode): | |
| try: | |
| if case and case != "": | |
| return [ | |
| tao_bieu_do_du_doan(case, chu_ky, mode), | |
| tao_thong_tin_tom_tat(case, chu_ky, mode) | |
| ] | |
| else: | |
| return [tao_bieu_do_mac_dinh(), tao_tom_tat_mac_dinh()] | |
| except: | |
| return [tao_bieu_do_mac_dinh(), tao_tom_tat_mac_dinh()] | |
| che_do_xem.change( | |
| fn=cap_nhat_khi_doi_che_do, | |
| inputs=[lua_chon_dropdown, chu_ky_dropdown, che_do_xem], | |
| outputs=[bieu_do_du_bao, hien_thi_tom_tat] | |
| ) | |
| # Tự động load case đầu tiên với biểu đồ mặc định | |
| demo.load( | |
| fn=lambda: [ | |
| cap_nhat_thong_tin_case(list(TRUONG_HOP_DEMO.keys())[0]), | |
| tao_bieu_do_mac_dinh(), | |
| tao_tom_tat_mac_dinh(), | |
| """# 🎯 CHIẾN LƯỢC ĐỀ XUẤT | |
| ## 📊 **Sẵn Sàng Phân Tích** | |
| **Nhấn 'Tạo Dự Báo AI' để xem phân tích chi tiết** | |
| Hệ thống AI sẽ tạo chiến lược đề xuất với: | |
| • **Phân tích xu hướng** thị trường | |
| • **Insights chi tiết** từ dữ liệu | |
| • **Khuyến nghị cụ thể** để tối ưa doanh thu | |
| • **Roadmap hành động** rõ ràng | |
| *🤖 Powered by FoxAI Intelligence*""" | |
| ], | |
| outputs=[hien_thi_thong_tin, bieu_do_du_bao, hien_thi_tom_tat, hien_thi_bao_cao] | |
| ) | |
| # Footer thông tin hệ thống với branding FoxAI | |
| gr.HTML(f""" | |
| <div class='info-card' style="margin-top: 40px; text-align: center;"> | |
| <!-- Logo Center --> | |
| <div style=" | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| margin-bottom: 35px; | |
| "> | |
| <img src="{FOXAI_LOGO_URL}" alt="FoxAI" style=" | |
| height: 80px; | |
| filter: drop-shadow(0 3px 6px rgba(0,0,0,0.25)); | |
| "/> | |
| </div> | |
| <!-- Grid --> | |
| <div style=" | |
| display: grid; | |
| grid-template-columns: repeat(auto-fit, minmax(280px, 1fr)); | |
| gap: 25px; | |
| margin: 30px 0; | |
| "> | |
| <div style=" | |
| padding: 25px; | |
| border-radius: 16px; | |
| background: rgba(255,255,255,0.05); | |
| border: 1px solid rgba(255,255,255,0.1); | |
| "> | |
| <h4 style="color: #fff; margin-bottom: 15px; font-size: 1.4em; font-weight: 700;"> | |
| 🎯 Tính Năng | |
| </h4> | |
| <div style="color: #f1f5f9; font-weight: 500; line-height: 1.8;"> | |
| ✨ AI Insights thông minh<br/> | |
| 📊 Multi-view linh hoạt<br/> | |
| 📈 Báo cáo chuyên nghiệp<br/> | |
| ⚡ Cập nhật real-time | |
| </div> | |
| </div> | |
| <div style=" | |
| padding: 25px; | |
| border-radius: 16px; | |
| background: rgba(255,255,255,0.05); | |
| border: 1px solid rgba(255,255,255,0.1); | |
| "> | |
| <h4 style="color: #fff; margin-bottom: 15px; font-size: 1.4em; font-weight: 700;"> | |
| 📊 Chất Lượng | |
| </h4> | |
| <div style="color: #f1f5f9; font-weight: 500; line-height: 1.8;"> | |
| 🎯 Độ chính xác cao<br/> | |
| 📅 Dự báo theo quý<br/> | |
| 🔄 Market intelligence<br/> | |
| 💼 Enterprise ready | |
| </div> | |
| </div> | |
| </div> | |
| <!-- Footer strip --> | |
| <div style=" | |
| margin-top: 25px; | |
| padding: 20px; | |
| border-radius: 16px; | |
| background: rgba(30,41,59,0.7); | |
| border: 1px solid rgba(255,255,255,0.1); | |
| "> | |
| <p style="color: #fff; font-size: 1.1em; font-weight: 600; margin: 0;"> | |
| 🦊 Powered by FoxAi • 🇻🇳 Made in Vietnam • 🤖 Advanced AI | |
| </p> | |
| <p style="color: #e2e8f0; margin-top: 8px; font-weight: 500;"> | |
| YOUR TRUSTED AI PARTNER | |
| </p> | |
| </div> | |
| </div> | |
| """) | |
| return demo | |
| # Chạy ứng dụng | |
| if __name__ == "__main__": | |
| demo = tao_ung_dung() | |
| demo.launch( | |
| share=True, | |
| server_name="0.0.0.0", | |
| server_port=7860, | |
| show_error=True | |
| ) | |