Sfoglia il codice sorgente

Version takes all pages of section : problem !

Abdenbi 1 mese fa
commit
dc64ce53b3
33 ha cambiato i file con 4948 aggiunte e 0 eliminazioni
  1. 14 0
      .gitignore
  2. 13 0
      04 - Scripts/.env
  3. 234 0
      04 - Scripts/Agents.py
  4. BIN
      04 - Scripts/Caisse_Générale_de_Prévoyance_(CGP)_2025/Rapport_S.02.xlsx
  5. BIN
      04 - Scripts/Caisse_Générale_de_Prévoyance_(CGP)_2025/Rapport_S.05.xlsx
  6. 58 0
      04 - Scripts/clean_DBSCAN.py
  7. 277 0
      04 - Scripts/function.py
  8. 89 0
      04 - Scripts/function_exctract.py
  9. 57 0
      04 - Scripts/main.py
  10. 6 0
      04 - Scripts/package-lock.json
  11. 16 0
      04 - Scripts/requirements.txt
  12. 72 0
      04 - Scripts/server.py
  13. 24 0
      04 - Scripts/sfcr-app/.gitignore
  14. 16 0
      04 - Scripts/sfcr-app/README.md
  15. 29 0
      04 - Scripts/sfcr-app/eslint.config.js
  16. 13 0
      04 - Scripts/sfcr-app/index.html
  17. 2647 0
      04 - Scripts/sfcr-app/package-lock.json
  18. 27 0
      04 - Scripts/sfcr-app/package.json
  19. 0 0
      04 - Scripts/sfcr-app/public/favicon.svg
  20. 24 0
      04 - Scripts/sfcr-app/public/icons.svg
  21. 97 0
      04 - Scripts/sfcr-app/src/App.css
  22. 457 0
      04 - Scripts/sfcr-app/src/App.jsx
  23. BIN
      04 - Scripts/sfcr-app/src/assets/hero.png
  24. 0 0
      04 - Scripts/sfcr-app/src/assets/react.svg
  25. 0 0
      04 - Scripts/sfcr-app/src/assets/vite.svg
  26. 43 0
      04 - Scripts/sfcr-app/src/constants.js
  27. 10 0
      04 - Scripts/sfcr-app/src/index.css
  28. 10 0
      04 - Scripts/sfcr-app/src/main.jsx
  29. 587 0
      04 - Scripts/sfcr-app/src/styles.js
  30. 7 0
      04 - Scripts/sfcr-app/vite.config.js
  31. 35 0
      04 - Scripts/test_luncher.py
  32. 42 0
      04 - Scripts/tools.py
  33. 44 0
      04 - Scripts/workflow_agents.py

+ 14 - 0
.gitignore

@@ -0,0 +1,14 @@
+# Image
+graph_workflow.png
+
+# Dossiers de données / projets
+01 - Sources/
+02 - Inputs/
+03 - Outputs/
+
+# VSCode
+.vscode/
+
+# Python cache
+__pycache__/
+*.pyc

+ 13 - 0
04 - Scripts/.env

@@ -0,0 +1,13 @@
+Project_final/.env 
+LANGFUSE_SECRET_KEY="sk-lf-da8a0214-de29-43e0-9c53-f85f8371a37f"
+LANGFUSE_PUBLIC_KEY="pk-lf-f07364b3-d0a6-4252-8d65-447500b27cb7"
+LANGFUSE_BASE_URL="https://cloud.langfuse.com"
+
+
+SERPER_API_KEY="5d34667463124d3a83b2b04987a3d80bb1dd3d00"
+GOOGLE_API_KEY="AIzaSyAdLpi0Z6Vh9wxjXY3qrdTKOJ7OvBRC4ZQ"
+GROQ_API_KEY ="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"
+OPENAI_API_KEY="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"
+MISTRAL_API_KEY="LrPwFxTOrrhORneWamW4VNP4UA0k7jPN"
+
+OPENROUTER_API_KEY="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"

+ 234 - 0
04 - Scripts/Agents.py

@@ -0,0 +1,234 @@
+import os 
+from typing import Annotated , Sequence , TypedDict , Optional , List , Any
+from langchain_core.messages import BaseMessage, HumanMessage, SystemMessage
+from langgraph.graph.message import add_messages
+from dotenv import load_dotenv  
+from langchain_openai import ChatOpenAI 
+from langchain_groq import ChatGroq 
+import pandas as pd
+from langfuse import get_client
+import json
+
+from tools import  excel_code_interpreter
+
+from paddleocr import PaddleOCR
+
+from function import (
+    preparer_image_zoom_hd, 
+    extraire_donnees_ocr, 
+    nettoyage_sortie_ocr, 
+    formater_donnees_section,
+    sauvegarder_fichier_unique,
+    to_points , 
+    build_lines,
+    cluster_lines , 
+    merge_close_lines
+
+)
+
+from clean_DBSCAN import (
+    transform_to_clean_markdown
+)
+
+
+
+load_dotenv() 
+
+
+tools = [excel_code_interpreter]
+
+
+class AgentState(TypedDict) : 
+    messages : Annotated[Sequence[BaseMessage],add_messages]
+    pdf_path : str 
+    pages : List[int]
+    entreprise_name : str 
+    section_name : str
+    markdown : str
+
+
+model_llama = ChatGroq(model="llama-3.3-70b-versatile") 
+
+model_openai = ChatOpenAI(model="gpt-4.1" , temperature=0.2) 
+
+model_ai = model_llama.bind_tools(tools)
+
+ocr = PaddleOCR(use_angle_cls=True, lang='fr', det_limit_side_len=10000, show_log=False)
+
+
+
+def agent_ocr(state: AgentState):
+    pdf_path = state.get("pdf_path")
+    pages = state.get("pages")
+
+    
+
+    texte_accumule = []
+    all_points = []
+
+    for page_index in pages:
+
+        page_index = page_index -1
+        
+        try:
+            img_finale = preparer_image_zoom_hd(pdf_path, page_index)
+            raw_data = extraire_donnees_ocr(img_finale, ocr)
+
+            data_propre = nettoyage_sortie_ocr(raw_data)
+
+            # 🔥 garder structure brute
+            points = to_points(data_propre)
+            all_points.extend(points)
+
+            # debug texte
+            lignes_page = formater_donnees_section(data_propre, page_index)
+            texte_accumule.extend(lignes_page)
+
+            print(f"✅ Page {page_index + 1} traitée.")
+
+        except Exception as e:
+            print(f"❌ Erreur Page {page_index + 1}: {e}")
+
+    # 🔥 DBSCAN GLOBAL (beaucoup plus robuste)
+    clusters = cluster_lines(all_points, eps=0.5)
+    lignes = build_lines(clusters)
+    lignes = merge_close_lines(lignes)
+
+    # 🔥 TEXTE pour LLM
+    contenu_final = "\n".join(texte_accumule)
+
+    message_ocr = HumanMessage(
+        content=f"Voici les données OCR brutes :\n{contenu_final}"
+    )
+  
+    markdown = transform_to_clean_markdown(lignes)
+    
+    
+
+    return {
+        "messages": [message_ocr],
+        "lignes" : lignes,
+        "markdown": markdown 
+    } 
+
+
+
+
+def agent_extracteur(state:AgentState) : 
+
+    prompt = """
+    Tu es un extracteur de tableaux QRT (Solvabilité II).
+
+    Ton but est de transcrire les coordonnées spatiales OCR en une hiérarchie logique de données.
+
+    Objectif : convertir le tableau en JSON sans perte d’information.
+
+    Règles :
+    - Utilise Rxxxx (lignes) et Cxxxx (colonnes) comme clés.
+    - Copie STRICTEMENT les valeurs, sans calcul ni modification.
+    - Supprime uniquement les espaces dans les nombres :
+    "3 297 388" → 3297388
+    - Ne tronque pas, n’arrondis pas, garde toute la précision.
+    - Garde les 0.
+    - Si pas de libellé pour un Rxxxx → "".
+    - Optimise la structure (par lignes ou colonnes) pour réduire la taille.
+
+    ⚠️ CONTRAINTE DE SORTIE (OBLIGATOIRE) :
+    - Réponds UNIQUEMENT avec un objet JSON valide
+    - NE PAS ajouter de texte avant ou après
+    - NE PAS utiliser ```json ou ```
+    - NE PAS commenter
+    - NE PAS expliquer
+
+    Sortie attendue (exemple strict) :
+    {
+    "labels": { "Rxxxx": "Rxxxx" },
+    "data": {
+        "Cxxxx": { "Rxxxx": 123456 }
+    }
+    }
+    """
+
+    msg = [
+    SystemMessage(content=prompt),
+    HumanMessage(content=json.dumps(state["markdown"]))
+     ]
+
+    response = model_llama.invoke(msg)
+
+    return{"messages" : [response]}
+
+
+
+
+
+def agent_builder(state: AgentState):
+    """
+    Agent Builder (Version non-IA) : 
+    Prend le JSON de l'extracteur et génère physiquement le fichier Excel.
+    """
+    print(f"🏗️ Construction du fichier Excel pour : {state['entreprise_name']}...")
+
+    try:
+        # 1. Récupération du JSON depuis le dernier message de l'extracteur
+        # On suppose que l'extracteur a renvoyé un dictionnaire ou une string JSON
+        import json
+        last_message_content = state["messages"][-1].content
+        
+        # Nettoyage si le LLM a mis des balises ```json
+        if "```json" in last_message_content:
+            last_message_content = last_message_content.split("```json")[1].split("```")[0].strip()
+        
+        payload = json.loads(last_message_content)
+        
+        labels = payload.get("labels", {})
+        data_json = payload.get("data", {})
+
+        # 2. Préparation de la structure du dossier
+        folder = state['entreprise_name'].replace(" ", "_")
+        if not os.path.exists(folder):
+            os.makedirs(folder)
+        
+        output_file = os.path.join(folder, f"Rapport_{state['section_name']}.xlsx")
+
+        # 3. Construction du DataFrame
+        # On récupère toutes les colonnes et toutes les lignes uniques
+        all_rows = list(labels.keys())
+        all_cols = list(data_json.keys())
+
+        # Création d'une matrice vide remplie de NaN
+        df = pd.DataFrame(index=all_rows, columns=all_cols)
+
+        # Remplissage des données
+        for col, row_values in data_json.items():
+            for row, val in row_values.items():
+                df.at[row, col] = val
+
+        # 4. Ajout de la colonne Libellé au début
+        #df.insert(0, "Libellé", df.index.map(labels))
+
+        # 5. Exportation avec formatage pro via XlsxWriter
+        with pd.ExcelWriter(output_file, engine="xlsxwriter") as writer:
+            df.to_excel(writer, sheet_name="QRT_Export", index=True)
+            
+            workbook  = writer.book
+            worksheet = writer.sheets["QRT_Export"]
+
+            # --- Ajout de formats ---
+            header_fmt = workbook.add_format({'bold': True, 'bg_color': '#1F3864', 'font_color': 'white', 'border': 1})
+            num_fmt    = workbook.add_format({'num_format': '#,##0', 'border': 1})
+            label_fmt  = workbook.add_format({'bg_color': '#D6E4F0', 'border': 1})
+
+            # Appliquer les formats aux colonnes
+            worksheet.set_column(1, 1, 60) # Libellé
+            worksheet.set_column(0, 0, 15) # Code R
+            
+            worksheet.set_column(2, len(df.columns)+1, 18, num_fmt) # Valeurs
+
+        success_msg = f"✓ Fichier Excel généré : {output_file} ({len(df)} lignes x {len(all_cols)} colonnes)"
+        return {"messages": [HumanMessage(content=success_msg)]}
+
+    except Exception as e:
+        error_msg = f"❌ Erreur lors de la construction Excel : {str(e)}"
+        print(error_msg)
+        return {"messages": [HumanMessage(content=error_msg)]}

BIN
04 - Scripts/Caisse_Générale_de_Prévoyance_(CGP)_2025/Rapport_S.02.xlsx


BIN
04 - Scripts/Caisse_Générale_de_Prévoyance_(CGP)_2025/Rapport_S.05.xlsx


+ 58 - 0
04 - Scripts/clean_DBSCAN.py

@@ -0,0 +1,58 @@
+
+import json
+
+
+def get_closest_col(x, current_cols):
+    """Trouve la colonne dont l'X est le plus proche de la valeur."""
+    return min(current_cols, key=lambda c: abs(x - c[0]))
+
+def transform_to_clean_markdown(data):
+    current_cols = [] 
+    output_lines = []
+    
+    for entry in data:
+        points = entry.get("points", [])
+        if not points: 
+            continue
+        
+        # 1. Détection et formatage des En-têtes (Cxxxx)
+        headers_in_row = [(p[0], p[2]) for p in points if str(p[2]).startswith('C')]
+        
+        if headers_in_row:
+            # On ajoute un séparateur si un tableau existait déjà avant
+            if output_lines:
+                output_lines.append("\n---\n")
+                
+            current_cols = headers_in_row
+            col_names = [c[1] for c in current_cols]
+            # Header propre sans coordonnées
+            output_lines.append(f"| Code | {' | '.join(col_names)} |")
+            output_lines.append(f"| :--- | {' | '.join([':---'] * len(col_names))} |")
+            continue
+
+        # 2. Détection et alignement des Données (Rxxxx)
+        row_label_pt = next((p for p in points if str(p[2]).startswith('R')), None)
+        
+        if row_label_pt and current_cols:
+            row_label = row_label_pt[2]
+            # Initialisation de la ligne avec des "0"
+            row_dict = {c[1]: "0" for c in current_cols}
+            
+            for p in points:
+                x_val, _, text = p
+                if text == row_label: 
+                    continue
+                
+                # Placement précis via coordonnée X
+                best_col = get_closest_col(x_val, current_cols)
+                row_dict[best_col[1]] = str(text).strip()
+            
+            # Construction de la ligne finale
+            ordered_values = [row_dict[c[1]] for c in current_cols]
+            output_lines.append(f"| **{row_label}** | {' | '.join(ordered_values)} |")
+            
+        # Note : On ignore volontairement le bloc "else" (INFO) pour nettoyer le bruit
+            
+    return "\n".join(output_lines)
+
+#print(transform_to_clean_markdown(json))

+ 277 - 0
04 - Scripts/function.py

@@ -0,0 +1,277 @@
+# utils.py
+import fitz
+import cv2
+import numpy as np
+import re
+
+def redresser_image_auto(img_array):
+    gray = cv2.cvtColor(img_array, cv2.COLOR_RGB2GRAY)
+    binary = cv2.adaptiveThreshold(gray, 255, cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY_INV, 15, 4)
+    h, w = binary.shape
+    kh = cv2.getStructuringElement(cv2.MORPH_RECT, (w // 10, 1))
+    kv = cv2.getStructuringElement(cv2.MORPH_RECT, (1, h // 10))
+    score_h = cv2.countNonZero(cv2.morphologyEx(binary, cv2.MORPH_OPEN, kh))
+    score_v = cv2.countNonZero(cv2.morphologyEx(binary, cv2.MORPH_OPEN, kv))
+    if (score_v / 1.5) > (score_h * 1.3):
+        return cv2.rotate(img_array, cv2.ROTATE_90_COUNTERCLOCKWISE)
+    return img_array
+
+def obtenir_zone_tableau_total(img_array):
+    gray = cv2.cvtColor(img_array, cv2.COLOR_RGB2GRAY)
+    blurred = cv2.GaussianBlur(gray, (5, 5), 0)
+    binary = cv2.adaptiveThreshold(blurred, 255, cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY_INV, 15, 4)
+    h, w = binary.shape
+    kernel = cv2.getStructuringElement(cv2.MORPH_RECT, (20, 5))
+    dilated = cv2.dilate(binary, kernel, iterations=3)
+    contours, _ = cv2.findContours(dilated, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
+    if not contours: return 0, h
+    y_points = []
+    for c in contours:
+        x, y, w_c, h_c = cv2.boundingRect(c)
+        if h_c > 10:
+            y_points.append(y); y_points.append(y + h_c)
+    if not y_points: return 0, h
+    return max(0, min(y_points) - 100), min(h, max(y_points) + 100)
+
+def preparer_image_zoom_hd(pdf_path, page_index):
+    """Gère le double passage pour extraire une image HD cadrée."""
+    doc = fitz.open(pdf_path)
+    page = doc.load_page(page_index)
+
+    print("page fitz",page)
+    
+    # 1. Localisation basse résolution
+    pix_low = page.get_pixmap(matrix=fitz.Matrix(1, 1))
+    img_low = np.frombuffer(pix_low.samples, dtype=np.uint8).reshape(pix_low.h, pix_low.w, 3)
+    img_low = redresser_image_auto(img_low)
+    y_min, y_max = obtenir_zone_tableau_total(img_low)
+    
+    # 2. Calcul du recadrage
+    h_low = img_low.shape[0]
+    y_start_pct = y_min / h_low
+    y_end_pct = y_max / h_low
+    
+    full_rect = page.rect
+    crop_rect = fitz.Rect(full_rect.x0, full_rect.y0 + (full_rect.height * y_start_pct),
+                          full_rect.x1, full_rect.y0 + (full_rect.height * y_end_pct))
+    
+    # 3. Rendu Haute Résolution (x4)
+    pix_high = page.get_pixmap(matrix=fitz.Matrix(4, 4), clip=crop_rect, colorspace=fitz.csRGB)
+    img_finale = np.frombuffer(pix_high.samples, dtype=np.uint8).reshape(pix_high.h, pix_high.w, 3)
+    img_finale = redresser_image_auto(img_finale)
+    
+    doc.close()
+    return img_finale
+
+def extraire_donnees_ocr(img, ocr_model):
+    """Lance l'OCR et structure les résultats par coordonnées."""
+    h_f, w_f = img.shape[:2]
+    result = ocr_model.ocr(img, cls=True)
+    extracted = []
+    if result and result[0]:
+        for line in result[0]:
+            box, (text, conf) = line[0], line[1]
+
+            if len(re.sub(r'[^a-zA-Z]', '', text)) > 2:
+             continue
+
+
+            if conf >= 0.6:
+                x_c, y_c = sum(p[0] for p in box) / 4, sum(p[1] for p in box) / 4
+                extracted.append({
+                    "text": text, 
+                    "x_pct": round(x_c / w_f * 100, 1),
+                    "y_pct": round(y_c / h_f * 100, 1), 
+                    "y_c": y_c,
+                      "x_c": x_c
+                })
+    extracted.sort(key=lambda r: (r["y_c"], r["x_c"]))
+    return extracted
+
+
+
+
+def nettoyer_texte_ocr(text):
+    if not text:
+        return ""
+
+    text = str(text)
+
+    # supprimer artefacts OCR fréquents
+    text = text.replace("]", "").replace("[", "").replace("/", "")
+
+    # corriger O → 0 uniquement si texte numérique
+    if re.match(r'^[\d\sO]+$', text):
+        text = text.replace("O", "0")
+
+    # corriger erreurs classiques R/RO
+    text = text.replace("RO", "R0")
+
+    # espaces propres
+    text = re.sub(r"\s+", " ", text)
+
+    return text.strip()
+
+def nettoyage_sortie_ocr(data):
+    """
+    Nettoie les données OCR pour DBSCAN :
+    - supprime lignes vides
+    - supprime artefacts inutiles
+    - garde textes utiles (mots, codes R/C, nombres)
+    """
+
+    cleaned = []
+
+    for r in data:
+        text = nettoyer_texte_ocr(r.get("text", ""))
+
+        # ❌ ignorer lignes totalement vides
+        if not text:
+            continue
+
+        # ❌ ignorer bruit pur (ex: "---", "...")
+        if re.match(r"^[\W_]+$", text):
+            continue
+
+        # ❌ ignorer "0" isolé (souvent bruit OCR)
+        if text == "0":
+            continue
+
+        # garder seulement si:
+        # - contient du texte utile OU
+        # - contient chiffre OU
+        if not re.search(r"[A-Za-z0-9]", text):
+            continue
+
+        cleaned.append({
+            "x_pct": float(r["x_pct"]),
+            "y_pct": float(r["y_pct"]),
+            "text": text
+        })
+
+    return cleaned
+
+def formater_donnees_section(data_page, page_index):
+    """
+    Format lisible + prêt à parser
+    """
+
+    lignes = [f"\n--- DONNÉES PAGE {page_index + 1} ---"]
+
+    for r in data_page:
+        x = float(r["x_pct"])
+        y = float(r["y_pct"])
+        text = r["text"]
+
+        lignes.append(f"x={x:.1f}% | y={y:.1f}% | {text}")
+
+    return lignes
+
+def to_points(data):
+    """
+    Convertit directement en format DBSCAN :
+    (x, y, text)
+    """
+    return [(r["x_pct"], r["y_pct"], r["text"]) for r in data]
+
+
+
+def sauvegarder_fichier_unique(contenu_total, pdf_path, section_name):
+    """Sauvegarde toutes les pages accumulées dans un seul fichier."""
+    # Nettoyage du nom de fichier
+    nom_propre = pdf_path.replace('.pdf', '').replace(' ', '_')
+    filename = f"{nom_propre}_{section_name}_complet.txt"
+    
+    with open(filename, "w", encoding="utf-8") as f:
+        f.write("\n".join(contenu_total))
+    
+    return filename
+
+
+
+########## DBSCAN 
+import numpy as np
+
+def prepare_for_dbscan(points):
+    """
+    points = [[x, y, text], ...]
+    """
+    coords = np.array([[p[1]] for p in points])  # 🔥 uniquement Y
+    return coords
+
+from sklearn.cluster import DBSCAN
+
+def cluster_lines(points, eps=0.4, min_samples=2):
+    coords = prepare_for_dbscan(points)
+
+    db = DBSCAN(eps=eps, min_samples=min_samples)
+    labels = db.fit_predict(coords)
+
+    clusters = {}
+    for label, point in zip(labels, points):
+        if label == -1:
+            continue  # bruit
+        clusters.setdefault(label, []).append(point)
+
+    return list(clusters.values())
+
+def build_lines(clusters):
+    lignes = []
+
+    for cluster in clusters:
+        # tri gauche → droite
+        cluster_sorted = sorted(cluster, key=lambda p: p[0])
+
+        texte = " ".join([p[2] for p in cluster_sorted])
+
+        lignes.append({
+            "y": np.mean([p[1] for p in cluster]),
+            "text": texte,
+            "points": cluster_sorted
+        })
+
+    # tri haut → bas
+    lignes = sorted(lignes, key=lambda l: l["y"])
+
+    return lignes
+
+
+def merge_close_lines(lignes, threshold=0.6):
+    merged = []
+    prev = None
+
+    for line in lignes:
+        if prev is None:
+            prev = line
+            continue
+
+        if abs(line["y"] - prev["y"]) < threshold:
+            prev["text"] += " " + line["text"]
+        else:
+            merged.append(prev)
+            prev = line
+
+    if prev:
+        merged.append(prev)
+
+    return merged
+
+
+def split_tables(points, eps_y=2.0):
+    """
+    Sépare les tableaux par distance verticale (Y)
+    """
+    import numpy as np
+    from sklearn.cluster import DBSCAN
+
+    y_coords = np.array([[p[1]] for p in points])  # فقط Y
+
+    clustering = DBSCAN(eps=eps_y, min_samples=5).fit(y_coords)
+
+    tables = {}
+    for label, point in zip(clustering.labels_, points):
+        if label == -1:
+            continue
+        tables.setdefault(label, []).append(point)
+
+    return list(tables.values())

+ 89 - 0
04 - Scripts/function_exctract.py

@@ -0,0 +1,89 @@
+import pandas as pd 
+import os
+import glob
+
+def extract_pdf_path(filename):
+    name_without_ext = filename.replace(".pdf", "")
+    parts = name_without_ext.rsplit("_", 1)
+    entreprise_name = parts[0]
+    année_extraction = parts[1]
+
+    # Chemin basé sur l'emplacement du fichier, pas le dossier courant
+    script_dir = os.path.dirname(os.path.abspath(__file__))
+    folder_abs = os.path.join(script_dir, "..", "01 - Sources", année_extraction)
+    folder_abs = os.path.abspath(folder_abs)
+
+    #print(f"CHEMIN : {folder_abs}")
+    #print(f"EXISTE : {os.path.exists(folder_abs)}")
+    #print(f"CONTENU : {os.listdir(folder_abs) if os.path.exists(folder_abs) else 'DOSSIER INTROUVABLE'}")
+
+    all_pdfs = glob.glob(os.path.join(folder_abs, "*.pdf"))
+
+    entreprise_lower = entreprise_name.strip().lower()
+    matches = [
+        p for p in all_pdfs
+        if os.path.basename(p).lower().startswith(entreprise_lower)
+    ]
+
+    if matches:
+        return matches[0]
+    else:
+        raise FileNotFoundError(
+            f"Aucun PDF trouvé pour '{entreprise_name}' en {année_extraction}\n"
+            f"PDFs disponibles :\n" +
+            "\n".join([os.path.basename(p) for p in all_pdfs])
+        )
+
+
+   
+
+def extract_pages(filename):
+    # Extrait entreprise_name et année depuis le nom du fichier
+    name_without_ext = filename.replace(".pdf", "")
+    parts = name_without_ext.rsplit("_", 1)
+    entreprise_name = parts[0]
+    année_extraction = int(parts[1])
+
+    # Chemin basé sur l'emplacement du fichier
+    script_dir = os.path.dirname(os.path.abspath(__file__))
+    excel_path = os.path.abspath(os.path.join(script_dir, "..", "02 - Inputs", "_QRTs_paramétrages_Abd.xlsx"))
+
+    # Lecture du fichier Excel
+    df = pd.read_excel(excel_path, sheet_name="Liste SFCR", header=3)
+    df.columns = [str(col).strip() for col in df.columns]
+
+    # Trouve la colonne Entité (insensible à l'encodage)
+    entite_col = [col for col in df.columns if "ntit" in col][0]
+
+    # Filtre la ligne correspondante
+    row = df[
+        (df[entite_col].str.strip().str.lower() == entreprise_name.strip().lower()) &
+        (df["Année"] == année_extraction)
+    ]
+
+    if row.empty:
+        raise ValueError(f"Aucune ligne trouvée pour '{entreprise_name}' en {année_extraction}")
+
+    row = row.iloc[0]
+
+    # Mapping colonnes Excel -> sections SFCR
+    section_columns = {
+        "S.02": "Pages S.02",
+        "S.05": "Pages S.05",
+        "S.12": "Pages S.12",
+        "S.22": "Pages S.22",
+        "S.25": "Pages S.25",
+        "S.28": "Pages S.28",
+    }
+
+    sections = {}
+    for section, col in section_columns.items():
+        val = row.get(col, None)
+        if pd.notna(val) and str(val).strip() != "":
+            # Convertit "135, 136" -> [135, 136]
+            pages = [int(p.strip()) for p in str(val).split(",")]
+            sections[section] = pages 
+        else:
+            sections[section] = []
+
+    return sections

+ 57 - 0
04 - Scripts/main.py

@@ -0,0 +1,57 @@
+from workflow_agents import app
+from function_exctract import extract_pages, extract_pdf_path
+from dotenv import load_dotenv
+from langfuse.langchain import CallbackHandler
+
+# Initialize Langfuse CallbackHandler for Langchain (tracing)
+langfuse_handler = CallbackHandler()
+
+load_dotenv()
+
+INPUTS = {
+    "Caisse Générale de Prévoyance (CGP)_2025.pdf": {
+        #"S.02": [],
+        "S.05": []
+    }
+}
+
+if __name__ == "__main__":
+    # 1. Boucle sur chaque entreprise (Fichier PDF)
+    for pdf_filename, sections_input in INPUTS.items():
+        print(f"\n DÉMARRAGE ENTREPRISE : {pdf_filename}")
+
+        entreprise_clean = pdf_filename.replace(".pdf", "")
+
+        # Récupère le vrai chemin du PDF
+        pdf_path = extract_pdf_path(pdf_filename)
+        print(f"   PDF trouvé : {pdf_path}")
+
+        # Récupère les pages depuis Excel
+        sections_with_pages = extract_pages(pdf_filename)
+
+        # Fusionne : garde seulement les sections choisies dans INPUTS
+        # et remplit les pages depuis Excel
+        sections = {
+            s: sections_with_pages.get(s, [])
+            for s in sections_input.keys()
+        }
+
+        # 2. Boucle sur chaque section
+        for section_name, pages in sections.items():
+            print(f"   Section : {section_name} — pages : {pages}")
+
+            initial_state = {
+                "messages": [],
+                "pdf_path": pdf_path,
+                "pages": pages,
+                "entreprise_name": entreprise_clean,
+                "section_name": section_name
+            }
+
+            try:
+                    final_state = app.invoke(initial_state, config={"callbacks": [langfuse_handler]})
+                    print(f"   Section {section_name} terminée.")
+            except Exception as e:
+                    print(f"  ❌ Erreur section {section_name}: {e}")
+
+    #print("\n✨ TOUTES LES ENTREPRISES ET SECTIONS ONT ÉTÉ TRAITÉES.")

+ 6 - 0
04 - Scripts/package-lock.json

@@ -0,0 +1,6 @@
+{
+  "name": "SFCR_extraction_tool",
+  "lockfileVersion": 3,
+  "requires": true,
+  "packages": {}
+}

+ 16 - 0
04 - Scripts/requirements.txt

@@ -0,0 +1,16 @@
+numpy==1.26.4
+paddlepaddle==2.6.2
+paddleocr==2.7.3
+xlsxwriter
+
+dotenv
+langchain-openai==0.2.5 
+langgraph==0.2.53
+langchain_groq
+langfuse==4.2.0
+
+langchain
+
+protobuf==3.20.3
+scikit-learn
+flask-cors-6.0.2

+ 72 - 0
04 - Scripts/server.py

@@ -0,0 +1,72 @@
+from flask import Flask, request, jsonify
+from flask_cors import CORS
+import subprocess
+import json
+import re
+import os
+
+app = Flask(__name__)
+CORS(app)
+
+import sys # Ajoute cet import en haut de ton fichier
+
+@app.route("/run-extraction", methods=["POST"])
+def run_extraction():
+    data = request.json
+    inputs = data.get("inputs", {})
+
+    script_dir = os.path.dirname(os.path.abspath(__file__))
+    main_path = os.path.join(script_dir, "main.py")
+
+    # 1. Mise à jour du fichier main.py
+    try:
+        with open(main_path, "r", encoding="utf-8") as f:
+            content = f.read()
+
+        # Nettoyage de l'ancien bloc INPUTS (ton regex/logique actuelle)
+        start_idx = content.find("INPUTS = {")
+        if start_idx != -1:
+            brace_count = 0
+            end_idx = start_idx
+            for i, char in enumerate(content[start_idx:]):
+                if char == "{": brace_count += 1
+                elif char == "}":
+                    brace_count -= 1
+                    if brace_count == 0:
+                        end_idx = start_idx + i + 1
+                        break
+            content = content[:start_idx] + content[end_idx:].lstrip()
+
+        # Insertion des nouveaux inputs
+        new_inputs_str = f"INPUTS = {json.dumps(inputs, indent=4, ensure_ascii=False)}\n\n"
+        content = new_inputs_str + content
+
+        with open(main_path, "w", encoding="utf-8") as f:
+            f.write(content)
+            f.flush() # Force l'écriture sur le disque
+            os.fsync(f.fileno()) # Sécurité supplémentaire pour Linux/Mac
+            
+    except Exception as e:
+        return jsonify({"stdout": "", "stderr": f"Erreur écriture : {str(e)}", "returncode": -1})
+
+    # 2. Exécution du script
+    try:
+        # Utilisation de sys.executable pour éviter les erreurs de chemin Python
+        result = subprocess.run(
+            [sys.executable, main_path], 
+            capture_output=True,
+            text=True,
+            cwd=script_dir,
+            timeout=60
+        )
+        
+        return jsonify({
+            "stdout": result.stdout,
+            "stderr": result.stderr,
+            "returncode": result.returncode
+        })
+
+    except subprocess.TimeoutExpired:
+        return jsonify({"stdout": "", "stderr": "Timeout dépassé (60s)", "returncode": -1})
+    except Exception as e:
+        return jsonify({"stdout": "", "stderr": f"Erreur exécution : {str(e)}", "returncode": -1})

+ 24 - 0
04 - Scripts/sfcr-app/.gitignore

@@ -0,0 +1,24 @@
+# Logs
+logs
+*.log
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+pnpm-debug.log*
+lerna-debug.log*
+
+node_modules
+dist
+dist-ssr
+*.local
+
+# Editor directories and files
+.vscode/*
+!.vscode/extensions.json
+.idea
+.DS_Store
+*.suo
+*.ntvs*
+*.njsproj
+*.sln
+*.sw?

+ 16 - 0
04 - Scripts/sfcr-app/README.md

@@ -0,0 +1,16 @@
+# React + Vite
+
+This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
+
+Currently, two official plugins are available:
+
+- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Oxc](https://oxc.rs)
+- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/)
+
+## React Compiler
+
+The React Compiler is not enabled on this template because of its impact on dev & build performances. To add it, see [this documentation](https://react.dev/learn/react-compiler/installation).
+
+## Expanding the ESLint configuration
+
+If you are developing a production application, we recommend using TypeScript with type-aware lint rules enabled. Check out the [TS template](https://github.com/vitejs/vite/tree/main/packages/create-vite/template-react-ts) for information on how to integrate TypeScript and [`typescript-eslint`](https://typescript-eslint.io) in your project.

+ 29 - 0
04 - Scripts/sfcr-app/eslint.config.js

@@ -0,0 +1,29 @@
+import js from '@eslint/js'
+import globals from 'globals'
+import reactHooks from 'eslint-plugin-react-hooks'
+import reactRefresh from 'eslint-plugin-react-refresh'
+import { defineConfig, globalIgnores } from 'eslint/config'
+
+export default defineConfig([
+  globalIgnores(['dist']),
+  {
+    files: ['**/*.{js,jsx}'],
+    extends: [
+      js.configs.recommended,
+      reactHooks.configs.flat.recommended,
+      reactRefresh.configs.vite,
+    ],
+    languageOptions: {
+      ecmaVersion: 2020,
+      globals: globals.browser,
+      parserOptions: {
+        ecmaVersion: 'latest',
+        ecmaFeatures: { jsx: true },
+        sourceType: 'module',
+      },
+    },
+    rules: {
+      'no-unused-vars': ['error', { varsIgnorePattern: '^[A-Z_]' }],
+    },
+  },
+])

+ 13 - 0
04 - Scripts/sfcr-app/index.html

@@ -0,0 +1,13 @@
+<!doctype html>
+<html lang="en">
+  <head>
+    <meta charset="UTF-8" />
+    <link rel="icon" type="image/svg+xml" href="/favicon.svg" />
+    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
+    <title>sfcr-app</title>
+  </head>
+  <body>
+    <div id="root"></div>
+    <script type="module" src="/src/main.jsx"></script>
+  </body>
+</html>

+ 2647 - 0
04 - Scripts/sfcr-app/package-lock.json

@@ -0,0 +1,2647 @@
+{
+  "name": "sfcr-app",
+  "version": "0.0.0",
+  "lockfileVersion": 3,
+  "requires": true,
+  "packages": {
+    "": {
+      "name": "sfcr-app",
+      "version": "0.0.0",
+      "dependencies": {
+        "react": "^19.2.5",
+        "react-dom": "^19.2.5"
+      },
+      "devDependencies": {
+        "@eslint/js": "^9.39.4",
+        "@types/react": "^19.2.14",
+        "@types/react-dom": "^19.2.3",
+        "@vitejs/plugin-react": "^6.0.1",
+        "eslint": "^9.39.4",
+        "eslint-plugin-react-hooks": "^7.1.1",
+        "eslint-plugin-react-refresh": "^0.5.2",
+        "globals": "^17.5.0",
+        "vite": "^8.0.9"
+      }
+    },
+    "node_modules/@babel/code-frame": {
+      "version": "7.29.0",
+      "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz",
+      "integrity": "sha512-9NhCeYjq9+3uxgdtp20LSiJXJvN0FeCtNGpJxuMFZ1Kv3cWUNb6DOhJwUvcVCzKGR66cw4njwM6hrJLqgOwbcw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/helper-validator-identifier": "^7.28.5",
+        "js-tokens": "^4.0.0",
+        "picocolors": "^1.1.1"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/compat-data": {
+      "version": "7.29.0",
+      "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.29.0.tgz",
+      "integrity": "sha512-T1NCJqT/j9+cn8fvkt7jtwbLBfLC/1y1c7NtCeXFRgzGTsafi68MRv8yzkYSapBnFA6L3U2VSc02ciDzoAJhJg==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/core": {
+      "version": "7.29.0",
+      "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.29.0.tgz",
+      "integrity": "sha512-CGOfOJqWjg2qW/Mb6zNsDm+u5vFQ8DxXfbM09z69p5Z6+mE1ikP2jUXw+j42Pf1XTYED2Rni5f95npYeuwMDQA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/code-frame": "^7.29.0",
+        "@babel/generator": "^7.29.0",
+        "@babel/helper-compilation-targets": "^7.28.6",
+        "@babel/helper-module-transforms": "^7.28.6",
+        "@babel/helpers": "^7.28.6",
+        "@babel/parser": "^7.29.0",
+        "@babel/template": "^7.28.6",
+        "@babel/traverse": "^7.29.0",
+        "@babel/types": "^7.29.0",
+        "@jridgewell/remapping": "^2.3.5",
+        "convert-source-map": "^2.0.0",
+        "debug": "^4.1.0",
+        "gensync": "^1.0.0-beta.2",
+        "json5": "^2.2.3",
+        "semver": "^6.3.1"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/babel"
+      }
+    },
+    "node_modules/@babel/generator": {
+      "version": "7.29.1",
+      "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.29.1.tgz",
+      "integrity": "sha512-qsaF+9Qcm2Qv8SRIMMscAvG4O3lJ0F1GuMo5HR/Bp02LopNgnZBC/EkbevHFeGs4ls/oPz9v+Bsmzbkbe+0dUw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/parser": "^7.29.0",
+        "@babel/types": "^7.29.0",
+        "@jridgewell/gen-mapping": "^0.3.12",
+        "@jridgewell/trace-mapping": "^0.3.28",
+        "jsesc": "^3.0.2"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-compilation-targets": {
+      "version": "7.28.6",
+      "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.28.6.tgz",
+      "integrity": "sha512-JYtls3hqi15fcx5GaSNL7SCTJ2MNmjrkHXg4FSpOA/grxK8KwyZ5bubHsCq8FXCkua6xhuaaBit+3b7+VZRfcA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/compat-data": "^7.28.6",
+        "@babel/helper-validator-option": "^7.27.1",
+        "browserslist": "^4.24.0",
+        "lru-cache": "^5.1.1",
+        "semver": "^6.3.1"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-globals": {
+      "version": "7.28.0",
+      "resolved": "https://registry.npmjs.org/@babel/helper-globals/-/helper-globals-7.28.0.tgz",
+      "integrity": "sha512-+W6cISkXFa1jXsDEdYA8HeevQT/FULhxzR99pxphltZcVaugps53THCeiWA8SguxxpSp3gKPiuYfSWopkLQ4hw==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-module-imports": {
+      "version": "7.28.6",
+      "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.28.6.tgz",
+      "integrity": "sha512-l5XkZK7r7wa9LucGw9LwZyyCUscb4x37JWTPz7swwFE/0FMQAGpiWUZn8u9DzkSBWEcK25jmvubfpw2dnAMdbw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/traverse": "^7.28.6",
+        "@babel/types": "^7.28.6"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-module-transforms": {
+      "version": "7.28.6",
+      "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.28.6.tgz",
+      "integrity": "sha512-67oXFAYr2cDLDVGLXTEABjdBJZ6drElUSI7WKp70NrpyISso3plG9SAGEF6y7zbha/wOzUByWWTJvEDVNIUGcA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/helper-module-imports": "^7.28.6",
+        "@babel/helper-validator-identifier": "^7.28.5",
+        "@babel/traverse": "^7.28.6"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      },
+      "peerDependencies": {
+        "@babel/core": "^7.0.0"
+      }
+    },
+    "node_modules/@babel/helper-string-parser": {
+      "version": "7.27.1",
+      "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.27.1.tgz",
+      "integrity": "sha512-qMlSxKbpRlAridDExk92nSobyDdpPijUq2DW6oDnUqd0iOGxmQjyqhMIihI9+zv4LPyZdRje2cavWPbCbWm3eA==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-validator-identifier": {
+      "version": "7.28.5",
+      "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.28.5.tgz",
+      "integrity": "sha512-qSs4ifwzKJSV39ucNjsvc6WVHs6b7S03sOh2OcHF9UHfVPqWWALUsNUVzhSBiItjRZoLHx7nIarVjqKVusUZ1Q==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helper-validator-option": {
+      "version": "7.27.1",
+      "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.27.1.tgz",
+      "integrity": "sha512-YvjJow9FxbhFFKDSuFnVCe2WxXk1zWc22fFePVNEaWJEu8IrZVlda6N0uHwzZrUM1il7NC9Mlp4MaJYbYd9JSg==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/helpers": {
+      "version": "7.29.2",
+      "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.29.2.tgz",
+      "integrity": "sha512-HoGuUs4sCZNezVEKdVcwqmZN8GoHirLUcLaYVNBK2J0DadGtdcqgr3BCbvH8+XUo4NGjNl3VOtSjEKNzqfFgKw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/template": "^7.28.6",
+        "@babel/types": "^7.29.0"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/parser": {
+      "version": "7.29.2",
+      "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.29.2.tgz",
+      "integrity": "sha512-4GgRzy/+fsBa72/RZVJmGKPmZu9Byn8o4MoLpmNe1m8ZfYnz5emHLQz3U4gLud6Zwl0RZIcgiLD7Uq7ySFuDLA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/types": "^7.29.0"
+      },
+      "bin": {
+        "parser": "bin/babel-parser.js"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/@babel/template": {
+      "version": "7.28.6",
+      "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz",
+      "integrity": "sha512-YA6Ma2KsCdGb+WC6UpBVFJGXL58MDA6oyONbjyF/+5sBgxY/dwkhLogbMT2GXXyU84/IhRw/2D1Os1B/giz+BQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/code-frame": "^7.28.6",
+        "@babel/parser": "^7.28.6",
+        "@babel/types": "^7.28.6"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/traverse": {
+      "version": "7.29.0",
+      "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.29.0.tgz",
+      "integrity": "sha512-4HPiQr0X7+waHfyXPZpWPfWL/J7dcN1mx9gL6WdQVMbPnF3+ZhSMs8tCxN7oHddJE9fhNE7+lxdnlyemKfJRuA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/code-frame": "^7.29.0",
+        "@babel/generator": "^7.29.0",
+        "@babel/helper-globals": "^7.28.0",
+        "@babel/parser": "^7.29.0",
+        "@babel/template": "^7.28.6",
+        "@babel/types": "^7.29.0",
+        "debug": "^4.3.1"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@babel/types": {
+      "version": "7.29.0",
+      "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.29.0.tgz",
+      "integrity": "sha512-LwdZHpScM4Qz8Xw2iKSzS+cfglZzJGvofQICy7W7v4caru4EaAmyUuO6BGrbyQ2mYV11W0U8j5mBhd14dd3B0A==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/helper-string-parser": "^7.27.1",
+        "@babel/helper-validator-identifier": "^7.28.5"
+      },
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/@emnapi/core": {
+      "version": "1.9.2",
+      "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.9.2.tgz",
+      "integrity": "sha512-UC+ZhH3XtczQYfOlu3lNEkdW/p4dsJ1r/bP7H8+rhao3TTTMO1ATq/4DdIi23XuGoFY+Cz0JmCbdVl0hz9jZcA==",
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "dependencies": {
+        "@emnapi/wasi-threads": "1.2.1",
+        "tslib": "^2.4.0"
+      }
+    },
+    "node_modules/@emnapi/runtime": {
+      "version": "1.9.2",
+      "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.9.2.tgz",
+      "integrity": "sha512-3U4+MIWHImeyu1wnmVygh5WlgfYDtyf0k8AbLhMFxOipihf6nrWC4syIm/SwEeec0mNSafiiNnMJwbza/Is6Lw==",
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "dependencies": {
+        "tslib": "^2.4.0"
+      }
+    },
+    "node_modules/@emnapi/wasi-threads": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/@emnapi/wasi-threads/-/wasi-threads-1.2.1.tgz",
+      "integrity": "sha512-uTII7OYF+/Mes/MrcIOYp5yOtSMLBWSIoLPpcgwipoiKbli6k322tcoFsxoIIxPDqW01SQGAgko4EzZi2BNv2w==",
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "dependencies": {
+        "tslib": "^2.4.0"
+      }
+    },
+    "node_modules/@eslint-community/eslint-utils": {
+      "version": "4.9.1",
+      "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz",
+      "integrity": "sha512-phrYmNiYppR7znFEdqgfWHXR6NCkZEK7hwWDHZUjit/2/U0r6XvkDl0SYnoM51Hq7FhCGdLDT6zxCCOY1hexsQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "eslint-visitor-keys": "^3.4.3"
+      },
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      },
+      "peerDependencies": {
+        "eslint": "^6.0.0 || ^7.0.0 || >=8.0.0"
+      }
+    },
+    "node_modules/@eslint-community/eslint-utils/node_modules/eslint-visitor-keys": {
+      "version": "3.4.3",
+      "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz",
+      "integrity": "sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": "^12.22.0 || ^14.17.0 || >=16.0.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/@eslint-community/regexpp": {
+      "version": "4.12.2",
+      "resolved": "https://registry.npmjs.org/@eslint-community/regexpp/-/regexpp-4.12.2.tgz",
+      "integrity": "sha512-EriSTlt5OC9/7SXkRSCAhfSxxoSUgBm33OH+IkwbdpgoqsSsUg7y3uh+IICI/Qg4BBWr3U2i39RpmycbxMq4ew==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": "^12.0.0 || ^14.0.0 || >=16.0.0"
+      }
+    },
+    "node_modules/@eslint/config-array": {
+      "version": "0.21.2",
+      "resolved": "https://registry.npmjs.org/@eslint/config-array/-/config-array-0.21.2.tgz",
+      "integrity": "sha512-nJl2KGTlrf9GjLimgIru+V/mzgSK0ABCDQRvxw5BjURL7WfH5uoWmizbH7QB6MmnMBd8cIC9uceWnezL1VZWWw==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "@eslint/object-schema": "^2.1.7",
+        "debug": "^4.3.1",
+        "minimatch": "^3.1.5"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      }
+    },
+    "node_modules/@eslint/config-helpers": {
+      "version": "0.4.2",
+      "resolved": "https://registry.npmjs.org/@eslint/config-helpers/-/config-helpers-0.4.2.tgz",
+      "integrity": "sha512-gBrxN88gOIf3R7ja5K9slwNayVcZgK6SOUORm2uBzTeIEfeVaIhOpCtTox3P6R7o2jLFwLFTLnC7kU/RGcYEgw==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "@eslint/core": "^0.17.0"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      }
+    },
+    "node_modules/@eslint/core": {
+      "version": "0.17.0",
+      "resolved": "https://registry.npmjs.org/@eslint/core/-/core-0.17.0.tgz",
+      "integrity": "sha512-yL/sLrpmtDaFEiUj1osRP4TI2MDz1AddJL+jZ7KSqvBuliN4xqYY54IfdN8qD8Toa6g1iloph1fxQNkjOxrrpQ==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "@types/json-schema": "^7.0.15"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      }
+    },
+    "node_modules/@eslint/eslintrc": {
+      "version": "3.3.5",
+      "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-3.3.5.tgz",
+      "integrity": "sha512-4IlJx0X0qftVsN5E+/vGujTRIFtwuLbNsVUe7TO6zYPDR1O6nFwvwhIKEKSrl6dZchmYBITazxKoUYOjdtjlRg==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "ajv": "^6.14.0",
+        "debug": "^4.3.2",
+        "espree": "^10.0.1",
+        "globals": "^14.0.0",
+        "ignore": "^5.2.0",
+        "import-fresh": "^3.2.1",
+        "js-yaml": "^4.1.1",
+        "minimatch": "^3.1.5",
+        "strip-json-comments": "^3.1.1"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/@eslint/eslintrc/node_modules/globals": {
+      "version": "14.0.0",
+      "resolved": "https://registry.npmjs.org/globals/-/globals-14.0.0.tgz",
+      "integrity": "sha512-oahGvuMGQlPw/ivIYBjVSrWAfWLBeku5tpPE2fOPLi+WHffIWbuh2tCjhyQhTBPMf5E9jDEH4FOmTYgYwbKwtQ==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=18"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/@eslint/js": {
+      "version": "9.39.4",
+      "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.39.4.tgz",
+      "integrity": "sha512-nE7DEIchvtiFTwBw4Lfbu59PG+kCofhjsKaCWzxTpt4lfRjRMqG6uMBzKXuEcyXhOHoUp9riAm7/aWYGhXZ9cw==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      },
+      "funding": {
+        "url": "https://eslint.org/donate"
+      }
+    },
+    "node_modules/@eslint/object-schema": {
+      "version": "2.1.7",
+      "resolved": "https://registry.npmjs.org/@eslint/object-schema/-/object-schema-2.1.7.tgz",
+      "integrity": "sha512-VtAOaymWVfZcmZbp6E2mympDIHvyjXs/12LqWYjVw6qjrfF+VK+fyG33kChz3nnK+SU5/NeHOqrTEHS8sXO3OA==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      }
+    },
+    "node_modules/@eslint/plugin-kit": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/@eslint/plugin-kit/-/plugin-kit-0.4.1.tgz",
+      "integrity": "sha512-43/qtrDUokr7LJqoF2c3+RInu/t4zfrpYdoSDfYyhg52rwLV6TnOvdG4fXm7IkSB3wErkcmJS9iEhjVtOSEjjA==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "@eslint/core": "^0.17.0",
+        "levn": "^0.4.1"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      }
+    },
+    "node_modules/@humanfs/core": {
+      "version": "0.19.2",
+      "resolved": "https://registry.npmjs.org/@humanfs/core/-/core-0.19.2.tgz",
+      "integrity": "sha512-UhXNm+CFMWcbChXywFwkmhqjs3PRCmcSa/hfBgLIb7oQ5HNb1wS0icWsGtSAUNgefHeI+eBrA8I1fxmbHsGdvA==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "@humanfs/types": "^0.15.0"
+      },
+      "engines": {
+        "node": ">=18.18.0"
+      }
+    },
+    "node_modules/@humanfs/node": {
+      "version": "0.16.8",
+      "resolved": "https://registry.npmjs.org/@humanfs/node/-/node-0.16.8.tgz",
+      "integrity": "sha512-gE1eQNZ3R++kTzFUpdGlpmy8kDZD/MLyHqDwqjkVQI0JMdI1D51sy1H958PNXYkM2rAac7e5/CnIKZrHtPh3BQ==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "dependencies": {
+        "@humanfs/core": "^0.19.2",
+        "@humanfs/types": "^0.15.0",
+        "@humanwhocodes/retry": "^0.4.0"
+      },
+      "engines": {
+        "node": ">=18.18.0"
+      }
+    },
+    "node_modules/@humanfs/types": {
+      "version": "0.15.0",
+      "resolved": "https://registry.npmjs.org/@humanfs/types/-/types-0.15.0.tgz",
+      "integrity": "sha512-ZZ1w0aoQkwuUuC7Yf+7sdeaNfqQiiLcSRbfI08oAxqLtpXQr9AIVX7Ay7HLDuiLYAaFPu8oBYNq/QIi9URHJ3Q==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": ">=18.18.0"
+      }
+    },
+    "node_modules/@humanwhocodes/module-importer": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz",
+      "integrity": "sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": ">=12.22"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/nzakas"
+      }
+    },
+    "node_modules/@humanwhocodes/retry": {
+      "version": "0.4.3",
+      "resolved": "https://registry.npmjs.org/@humanwhocodes/retry/-/retry-0.4.3.tgz",
+      "integrity": "sha512-bV0Tgo9K4hfPCek+aMAn81RppFKv2ySDQeMoSZuvTASywNTnVJCArCZE2FWqpvIatKu7VMRLWlR1EazvVhDyhQ==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": ">=18.18"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/nzakas"
+      }
+    },
+    "node_modules/@jridgewell/gen-mapping": {
+      "version": "0.3.13",
+      "resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
+      "integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@jridgewell/sourcemap-codec": "^1.5.0",
+        "@jridgewell/trace-mapping": "^0.3.24"
+      }
+    },
+    "node_modules/@jridgewell/remapping": {
+      "version": "2.3.5",
+      "resolved": "https://registry.npmjs.org/@jridgewell/remapping/-/remapping-2.3.5.tgz",
+      "integrity": "sha512-LI9u/+laYG4Ds1TDKSJW2YPrIlcVYOwi2fUC6xB43lueCjgxV4lffOCZCtYFiH6TNOX+tQKXx97T4IKHbhyHEQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@jridgewell/gen-mapping": "^0.3.5",
+        "@jridgewell/trace-mapping": "^0.3.24"
+      }
+    },
+    "node_modules/@jridgewell/resolve-uri": {
+      "version": "3.1.2",
+      "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
+      "integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/@jridgewell/sourcemap-codec": {
+      "version": "1.5.5",
+      "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
+      "integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/@jridgewell/trace-mapping": {
+      "version": "0.3.31",
+      "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
+      "integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@jridgewell/resolve-uri": "^3.1.0",
+        "@jridgewell/sourcemap-codec": "^1.4.14"
+      }
+    },
+    "node_modules/@napi-rs/wasm-runtime": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/@napi-rs/wasm-runtime/-/wasm-runtime-1.1.4.tgz",
+      "integrity": "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow==",
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "dependencies": {
+        "@tybys/wasm-util": "^0.10.1"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/Brooooooklyn"
+      },
+      "peerDependencies": {
+        "@emnapi/core": "^1.7.1",
+        "@emnapi/runtime": "^1.7.1"
+      }
+    },
+    "node_modules/@oxc-project/types": {
+      "version": "0.126.0",
+      "resolved": "https://registry.npmjs.org/@oxc-project/types/-/types-0.126.0.tgz",
+      "integrity": "sha512-oGfVtjAgwQVVpfBrbtk4e1XDyWHRFta6BS3GWVzrF8xYBT2VGQAk39yJS/wFSMrZqoiCU4oghT3Ch0HaHGIHcQ==",
+      "dev": true,
+      "license": "MIT",
+      "funding": {
+        "url": "https://github.com/sponsors/Boshen"
+      }
+    },
+    "node_modules/@rolldown/binding-android-arm64": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.0-rc.16.tgz",
+      "integrity": "sha512-rhY3k7Bsae9qQfOtph2Pm2jZEA+s8Gmjoz4hhmx70K9iMQ/ddeae+xhRQcM5IuVx5ry1+bGfkvMn7D6MJggVSA==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "android"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-darwin-arm64": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-arm64/-/binding-darwin-arm64-1.0.0-rc.16.tgz",
+      "integrity": "sha512-rNz0yK078yrNn3DrdgN+PKiMOW8HfQ92jQiXxwX8yW899ayV00MLVdaCNeVBhG/TbH3ouYVObo8/yrkiectkcQ==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-darwin-x64": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-darwin-x64/-/binding-darwin-x64-1.0.0-rc.16.tgz",
+      "integrity": "sha512-r/OmdR00HmD4i79Z//xO06uEPOq5hRXdhw7nzkxQxwSavs3PSHa1ijntdpOiZ2mzOQ3fVVu8C1M19FoNM+dMUQ==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-freebsd-x64": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-freebsd-x64/-/binding-freebsd-x64-1.0.0-rc.16.tgz",
+      "integrity": "sha512-KcRE5w8h0OnjUatG8pldyD14/CQ5Phs1oxfR+3pKDjboHRo9+MkqQaiIZlZRpsxC15paeXme/I127tUa9TXJ6g==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "freebsd"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-arm-gnueabihf": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm-gnueabihf/-/binding-linux-arm-gnueabihf-1.0.0-rc.16.tgz",
+      "integrity": "sha512-bT0guA1bpxEJ/ZhTRniQf7rNF8ybvXOuWbNIeLABaV5NGjx4EtOWBTSRGWFU9ZWVkPOZ+HNFP8RMcBokBiZ0Kg==",
+      "cpu": [
+        "arm"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-arm64-gnu": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-gnu/-/binding-linux-arm64-gnu-1.0.0-rc.16.tgz",
+      "integrity": "sha512-+tHktCHWV8BDQSjemUqm/Jl/TPk3QObCTIjmdDy/nlupcujZghmKK2962LYrqFpWu+ai01AN/REOH3NEpqvYQg==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "libc": [
+        "glibc"
+      ],
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-arm64-musl": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-arm64-musl/-/binding-linux-arm64-musl-1.0.0-rc.16.tgz",
+      "integrity": "sha512-3fPzdREH806oRLxpTWW1Gt4tQHs0TitZFOECB2xzCFLPKnSOy90gwA7P29cksYilFO6XVRY1kzga0cL2nRjKPg==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "libc": [
+        "musl"
+      ],
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-ppc64-gnu": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-ppc64-gnu/-/binding-linux-ppc64-gnu-1.0.0-rc.16.tgz",
+      "integrity": "sha512-EKwI1tSrLs7YVw+JPJT/G2dJQ1jl9qlTTTEG0V2Ok/RdOenRfBw2PQdLPyjhIu58ocdBfP7vIRN/pvMsPxs/AQ==",
+      "cpu": [
+        "ppc64"
+      ],
+      "dev": true,
+      "libc": [
+        "glibc"
+      ],
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-s390x-gnu": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-s390x-gnu/-/binding-linux-s390x-gnu-1.0.0-rc.16.tgz",
+      "integrity": "sha512-Uknladnb3Sxqu6SEcqBldQyJUpk8NleooZEc0MbRBJ4inEhRYWZX0NJu12vNf2mqAq7gsofAxHrGghiUYjhaLQ==",
+      "cpu": [
+        "s390x"
+      ],
+      "dev": true,
+      "libc": [
+        "glibc"
+      ],
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-x64-gnu": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-gnu/-/binding-linux-x64-gnu-1.0.0-rc.16.tgz",
+      "integrity": "sha512-FIb8+uG49sZBtLTn+zt1AJ20TqVcqWeSIyoVt0or7uAWesgKaHbiBh6OpA/k9v0LTt+PTrb1Lao133kP4uVxkg==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "libc": [
+        "glibc"
+      ],
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-linux-x64-musl": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-linux-x64-musl/-/binding-linux-x64-musl-1.0.0-rc.16.tgz",
+      "integrity": "sha512-RuERhF9/EgWxZEXYWCOaViUWHIboceK4/ivdtQ3R0T44NjLkIIlGIAVAuCddFxsZ7vnRHtNQUrt2vR2n2slB2w==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "libc": [
+        "musl"
+      ],
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-openharmony-arm64": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-openharmony-arm64/-/binding-openharmony-arm64-1.0.0-rc.16.tgz",
+      "integrity": "sha512-mXcXnvd9GpazCxeUCCnZ2+YF7nut+ZOEbE4GtaiPtyY6AkhZWbK70y1KK3j+RDhjVq5+U8FySkKRb/+w0EeUwA==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "openharmony"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-wasm32-wasi": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-wasm32-wasi/-/binding-wasm32-wasi-1.0.0-rc.16.tgz",
+      "integrity": "sha512-3Q2KQxnC8IJOLqXmUMoYwyIPZU9hzRbnHaoV3Euz+VVnjZKcY8ktnNP8T9R4/GGQtb27C/UYKABxesKWb8lsvQ==",
+      "cpu": [
+        "wasm32"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "dependencies": {
+        "@emnapi/core": "1.9.2",
+        "@emnapi/runtime": "1.9.2",
+        "@napi-rs/wasm-runtime": "^1.1.4"
+      },
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-win32-arm64-msvc": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-arm64-msvc/-/binding-win32-arm64-msvc-1.0.0-rc.16.tgz",
+      "integrity": "sha512-tj7XRemQcOcFwv7qhpUxMTBbI5mWMlE4c1Omhg5+h8GuLXzyj8HviYgR+bB2DMDgRqUE+jiDleqSCRjx4aYk/Q==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/binding-win32-x64-msvc": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/binding-win32-x64-msvc/-/binding-win32-x64-msvc-1.0.0-rc.16.tgz",
+      "integrity": "sha512-PH5DRZT+F4f2PTXRXR8uJxnBq2po/xFtddyabTJVJs/ZYVHqXPEgNIr35IHTEa6bpa0Q8Awg+ymkTaGnKITw4g==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      }
+    },
+    "node_modules/@rolldown/pluginutils": {
+      "version": "1.0.0-rc.7",
+      "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-rc.7.tgz",
+      "integrity": "sha512-qujRfC8sFVInYSPPMLQByRh7zhwkGFS4+tyMQ83srV1qrxL4g8E2tyxVVyxd0+8QeBM1mIk9KbWxkegRr76XzA==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/@tybys/wasm-util": {
+      "version": "0.10.1",
+      "resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.1.tgz",
+      "integrity": "sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg==",
+      "dev": true,
+      "license": "MIT",
+      "optional": true,
+      "dependencies": {
+        "tslib": "^2.4.0"
+      }
+    },
+    "node_modules/@types/estree": {
+      "version": "1.0.8",
+      "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
+      "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/@types/json-schema": {
+      "version": "7.0.15",
+      "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
+      "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/@types/react": {
+      "version": "19.2.14",
+      "resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.14.tgz",
+      "integrity": "sha512-ilcTH/UniCkMdtexkoCN0bI7pMcJDvmQFPvuPvmEaYA/NSfFTAgdUSLAoVjaRJm7+6PvcM+q1zYOwS4wTYMF9w==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "csstype": "^3.2.2"
+      }
+    },
+    "node_modules/@types/react-dom": {
+      "version": "19.2.3",
+      "resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz",
+      "integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==",
+      "dev": true,
+      "license": "MIT",
+      "peerDependencies": {
+        "@types/react": "^19.2.0"
+      }
+    },
+    "node_modules/@vitejs/plugin-react": {
+      "version": "6.0.1",
+      "resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.1.tgz",
+      "integrity": "sha512-l9X/E3cDb+xY3SWzlG1MOGt2usfEHGMNIaegaUGFsLkb3RCn/k8/TOXBcab+OndDI4TBtktT8/9BwwW8Vi9KUQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@rolldown/pluginutils": "1.0.0-rc.7"
+      },
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      },
+      "peerDependencies": {
+        "@rolldown/plugin-babel": "^0.1.7 || ^0.2.0",
+        "babel-plugin-react-compiler": "^1.0.0",
+        "vite": "^8.0.0"
+      },
+      "peerDependenciesMeta": {
+        "@rolldown/plugin-babel": {
+          "optional": true
+        },
+        "babel-plugin-react-compiler": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/acorn": {
+      "version": "8.16.0",
+      "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz",
+      "integrity": "sha512-UVJyE9MttOsBQIDKw1skb9nAwQuR5wuGD3+82K6JgJlm/Y+KI92oNsMNGZCYdDsVtRHSak0pcV5Dno5+4jh9sw==",
+      "dev": true,
+      "license": "MIT",
+      "bin": {
+        "acorn": "bin/acorn"
+      },
+      "engines": {
+        "node": ">=0.4.0"
+      }
+    },
+    "node_modules/acorn-jsx": {
+      "version": "5.3.2",
+      "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz",
+      "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==",
+      "dev": true,
+      "license": "MIT",
+      "peerDependencies": {
+        "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
+      }
+    },
+    "node_modules/ajv": {
+      "version": "6.14.0",
+      "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.14.0.tgz",
+      "integrity": "sha512-IWrosm/yrn43eiKqkfkHis7QioDleaXQHdDVPKg0FSwwd/DuvyX79TZnFOnYpB7dcsFAMmtFztZuXPDvSePkFw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "fast-deep-equal": "^3.1.1",
+        "fast-json-stable-stringify": "^2.0.0",
+        "json-schema-traverse": "^0.4.1",
+        "uri-js": "^4.2.2"
+      },
+      "funding": {
+        "type": "github",
+        "url": "https://github.com/sponsors/epoberezkin"
+      }
+    },
+    "node_modules/ansi-styles": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz",
+      "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "color-convert": "^2.0.1"
+      },
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/ansi-styles?sponsor=1"
+      }
+    },
+    "node_modules/argparse": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz",
+      "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==",
+      "dev": true,
+      "license": "Python-2.0"
+    },
+    "node_modules/balanced-match": {
+      "version": "1.0.2",
+      "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+      "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/baseline-browser-mapping": {
+      "version": "2.10.20",
+      "resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.20.tgz",
+      "integrity": "sha512-1AaXxEPfXT+GvTBJFuy4yXVHWJBXa4OdbIebGN/wX5DlsIkU0+wzGnd2lOzokSk51d5LUmqjgBLRLlypLUqInQ==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "bin": {
+        "baseline-browser-mapping": "dist/cli.cjs"
+      },
+      "engines": {
+        "node": ">=6.0.0"
+      }
+    },
+    "node_modules/brace-expansion": {
+      "version": "1.1.14",
+      "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.14.tgz",
+      "integrity": "sha512-MWPGfDxnyzKU7rNOW9SP/c50vi3xrmrua/+6hfPbCS2ABNWfx24vPidzvC7krjU/RTo235sV776ymlsMtGKj8g==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "balanced-match": "^1.0.0",
+        "concat-map": "0.0.1"
+      }
+    },
+    "node_modules/browserslist": {
+      "version": "4.28.2",
+      "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz",
+      "integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "opencollective",
+          "url": "https://opencollective.com/browserslist"
+        },
+        {
+          "type": "tidelift",
+          "url": "https://tidelift.com/funding/github/npm/browserslist"
+        },
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/ai"
+        }
+      ],
+      "license": "MIT",
+      "dependencies": {
+        "baseline-browser-mapping": "^2.10.12",
+        "caniuse-lite": "^1.0.30001782",
+        "electron-to-chromium": "^1.5.328",
+        "node-releases": "^2.0.36",
+        "update-browserslist-db": "^1.2.3"
+      },
+      "bin": {
+        "browserslist": "cli.js"
+      },
+      "engines": {
+        "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
+      }
+    },
+    "node_modules/callsites": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz",
+      "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/caniuse-lite": {
+      "version": "1.0.30001790",
+      "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001790.tgz",
+      "integrity": "sha512-bOoxfJPyYo+ds6W0YfptaCWbFnJYjh2Y1Eow5lRv+vI2u8ganPZqNm1JwNh0t2ELQCqIWg4B3dWEusgAmsoyOw==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "opencollective",
+          "url": "https://opencollective.com/browserslist"
+        },
+        {
+          "type": "tidelift",
+          "url": "https://tidelift.com/funding/github/npm/caniuse-lite"
+        },
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/ai"
+        }
+      ],
+      "license": "CC-BY-4.0"
+    },
+    "node_modules/chalk": {
+      "version": "4.1.2",
+      "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz",
+      "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "ansi-styles": "^4.1.0",
+        "supports-color": "^7.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/chalk/chalk?sponsor=1"
+      }
+    },
+    "node_modules/color-convert": {
+      "version": "2.0.1",
+      "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
+      "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "color-name": "~1.1.4"
+      },
+      "engines": {
+        "node": ">=7.0.0"
+      }
+    },
+    "node_modules/color-name": {
+      "version": "1.1.4",
+      "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
+      "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/concat-map": {
+      "version": "0.0.1",
+      "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+      "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/convert-source-map": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
+      "integrity": "sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/cross-spawn": {
+      "version": "7.0.6",
+      "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
+      "integrity": "sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "path-key": "^3.1.0",
+        "shebang-command": "^2.0.0",
+        "which": "^2.0.1"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/csstype": {
+      "version": "3.2.3",
+      "resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
+      "integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/debug": {
+      "version": "4.4.3",
+      "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
+      "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "ms": "^2.1.3"
+      },
+      "engines": {
+        "node": ">=6.0"
+      },
+      "peerDependenciesMeta": {
+        "supports-color": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/deep-is": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
+      "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/detect-libc": {
+      "version": "2.1.2",
+      "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
+      "integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/electron-to-chromium": {
+      "version": "1.5.343",
+      "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.343.tgz",
+      "integrity": "sha512-YHnQ3MXI08icvL9ZKnEBy05F2EQ8ob01UaMOuMbM8l+4UcAq6MPPbBTJBbsBUg3H8JeZNt+O4fjsoWth3p6IFg==",
+      "dev": true,
+      "license": "ISC"
+    },
+    "node_modules/escalade": {
+      "version": "3.2.0",
+      "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
+      "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/escape-string-regexp": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz",
+      "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/eslint": {
+      "version": "9.39.4",
+      "resolved": "https://registry.npmjs.org/eslint/-/eslint-9.39.4.tgz",
+      "integrity": "sha512-XoMjdBOwe/esVgEvLmNsD3IRHkm7fbKIUGvrleloJXUZgDHig2IPWNniv+GwjyJXzuNqVjlr5+4yVUZjycJwfQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@eslint-community/eslint-utils": "^4.8.0",
+        "@eslint-community/regexpp": "^4.12.1",
+        "@eslint/config-array": "^0.21.2",
+        "@eslint/config-helpers": "^0.4.2",
+        "@eslint/core": "^0.17.0",
+        "@eslint/eslintrc": "^3.3.5",
+        "@eslint/js": "9.39.4",
+        "@eslint/plugin-kit": "^0.4.1",
+        "@humanfs/node": "^0.16.6",
+        "@humanwhocodes/module-importer": "^1.0.1",
+        "@humanwhocodes/retry": "^0.4.2",
+        "@types/estree": "^1.0.6",
+        "ajv": "^6.14.0",
+        "chalk": "^4.0.0",
+        "cross-spawn": "^7.0.6",
+        "debug": "^4.3.2",
+        "escape-string-regexp": "^4.0.0",
+        "eslint-scope": "^8.4.0",
+        "eslint-visitor-keys": "^4.2.1",
+        "espree": "^10.4.0",
+        "esquery": "^1.5.0",
+        "esutils": "^2.0.2",
+        "fast-deep-equal": "^3.1.3",
+        "file-entry-cache": "^8.0.0",
+        "find-up": "^5.0.0",
+        "glob-parent": "^6.0.2",
+        "ignore": "^5.2.0",
+        "imurmurhash": "^0.1.4",
+        "is-glob": "^4.0.0",
+        "json-stable-stringify-without-jsonify": "^1.0.1",
+        "lodash.merge": "^4.6.2",
+        "minimatch": "^3.1.5",
+        "natural-compare": "^1.4.0",
+        "optionator": "^0.9.3"
+      },
+      "bin": {
+        "eslint": "bin/eslint.js"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      },
+      "funding": {
+        "url": "https://eslint.org/donate"
+      },
+      "peerDependencies": {
+        "jiti": "*"
+      },
+      "peerDependenciesMeta": {
+        "jiti": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/eslint-plugin-react-hooks": {
+      "version": "7.1.1",
+      "resolved": "https://registry.npmjs.org/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-7.1.1.tgz",
+      "integrity": "sha512-f2I7Gw6JbvCexzIInuSbZpfdQ44D7iqdWX01FKLvrPgqxoE7oMj8clOfto8U6vYiz4yd5oKu39rRSVOe1zRu0g==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@babel/core": "^7.24.4",
+        "@babel/parser": "^7.24.4",
+        "hermes-parser": "^0.25.1",
+        "zod": "^3.25.0 || ^4.0.0",
+        "zod-validation-error": "^3.5.0 || ^4.0.0"
+      },
+      "engines": {
+        "node": ">=18"
+      },
+      "peerDependencies": {
+        "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0 || ^10.0.0"
+      }
+    },
+    "node_modules/eslint-plugin-react-refresh": {
+      "version": "0.5.2",
+      "resolved": "https://registry.npmjs.org/eslint-plugin-react-refresh/-/eslint-plugin-react-refresh-0.5.2.tgz",
+      "integrity": "sha512-hmgTH57GfzoTFjVN0yBwTggnsVUF2tcqi7RJZHqi9lIezSs4eFyAMktA68YD4r5kNw1mxyY4dmkyoFDb3FIqrA==",
+      "dev": true,
+      "license": "MIT",
+      "peerDependencies": {
+        "eslint": "^9 || ^10"
+      }
+    },
+    "node_modules/eslint-scope": {
+      "version": "8.4.0",
+      "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-8.4.0.tgz",
+      "integrity": "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg==",
+      "dev": true,
+      "license": "BSD-2-Clause",
+      "dependencies": {
+        "esrecurse": "^4.3.0",
+        "estraverse": "^5.2.0"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/eslint-visitor-keys": {
+      "version": "4.2.1",
+      "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-4.2.1.tgz",
+      "integrity": "sha512-Uhdk5sfqcee/9H/rCOJikYz67o0a2Tw2hGRPOG2Y1R2dg7brRe1uG0yaNQDHu+TO/uQPF/5eCapvYSmHUjt7JQ==",
+      "dev": true,
+      "license": "Apache-2.0",
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/espree": {
+      "version": "10.4.0",
+      "resolved": "https://registry.npmjs.org/espree/-/espree-10.4.0.tgz",
+      "integrity": "sha512-j6PAQ2uUr79PZhBjP5C5fhl8e39FmRnOjsD5lGnWrFU8i2G776tBK7+nP8KuQUTTyAZUwfQqXAgrVH5MbH9CYQ==",
+      "dev": true,
+      "license": "BSD-2-Clause",
+      "dependencies": {
+        "acorn": "^8.15.0",
+        "acorn-jsx": "^5.3.2",
+        "eslint-visitor-keys": "^4.2.1"
+      },
+      "engines": {
+        "node": "^18.18.0 || ^20.9.0 || >=21.1.0"
+      },
+      "funding": {
+        "url": "https://opencollective.com/eslint"
+      }
+    },
+    "node_modules/esquery": {
+      "version": "1.7.0",
+      "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.7.0.tgz",
+      "integrity": "sha512-Ap6G0WQwcU/LHsvLwON1fAQX9Zp0A2Y6Y/cJBl9r/JbW90Zyg4/zbG6zzKa2OTALELarYHmKu0GhpM5EO+7T0g==",
+      "dev": true,
+      "license": "BSD-3-Clause",
+      "dependencies": {
+        "estraverse": "^5.1.0"
+      },
+      "engines": {
+        "node": ">=0.10"
+      }
+    },
+    "node_modules/esrecurse": {
+      "version": "4.3.0",
+      "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz",
+      "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==",
+      "dev": true,
+      "license": "BSD-2-Clause",
+      "dependencies": {
+        "estraverse": "^5.2.0"
+      },
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/estraverse": {
+      "version": "5.3.0",
+      "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz",
+      "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==",
+      "dev": true,
+      "license": "BSD-2-Clause",
+      "engines": {
+        "node": ">=4.0"
+      }
+    },
+    "node_modules/esutils": {
+      "version": "2.0.3",
+      "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
+      "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
+      "dev": true,
+      "license": "BSD-2-Clause",
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/fast-deep-equal": {
+      "version": "3.1.3",
+      "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
+      "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/fast-json-stable-stringify": {
+      "version": "2.1.0",
+      "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz",
+      "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/fast-levenshtein": {
+      "version": "2.0.6",
+      "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz",
+      "integrity": "sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/fdir": {
+      "version": "6.5.0",
+      "resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
+      "integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=12.0.0"
+      },
+      "peerDependencies": {
+        "picomatch": "^3 || ^4"
+      },
+      "peerDependenciesMeta": {
+        "picomatch": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/file-entry-cache": {
+      "version": "8.0.0",
+      "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-8.0.0.tgz",
+      "integrity": "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "flat-cache": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=16.0.0"
+      }
+    },
+    "node_modules/find-up": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz",
+      "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "locate-path": "^6.0.0",
+        "path-exists": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/flat-cache": {
+      "version": "4.0.1",
+      "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-4.0.1.tgz",
+      "integrity": "sha512-f7ccFPK3SXFHpx15UIGyRJ/FJQctuKZ0zVuN3frBo4HnK3cay9VEW0R6yPYFHC0AgqhukPzKjq22t5DmAyqGyw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "flatted": "^3.2.9",
+        "keyv": "^4.5.4"
+      },
+      "engines": {
+        "node": ">=16"
+      }
+    },
+    "node_modules/flatted": {
+      "version": "3.4.2",
+      "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.4.2.tgz",
+      "integrity": "sha512-PjDse7RzhcPkIJwy5t7KPWQSZ9cAbzQXcafsetQoD7sOJRQlGikNbx7yZp2OotDnJyrDcbyRq3Ttb18iYOqkxA==",
+      "dev": true,
+      "license": "ISC"
+    },
+    "node_modules/fsevents": {
+      "version": "2.3.3",
+      "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
+      "integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
+      "dev": true,
+      "hasInstallScript": true,
+      "license": "MIT",
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": "^8.16.0 || ^10.6.0 || >=11.0.0"
+      }
+    },
+    "node_modules/gensync": {
+      "version": "1.0.0-beta.2",
+      "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz",
+      "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6.9.0"
+      }
+    },
+    "node_modules/glob-parent": {
+      "version": "6.0.2",
+      "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
+      "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
+      "dev": true,
+      "license": "ISC",
+      "dependencies": {
+        "is-glob": "^4.0.3"
+      },
+      "engines": {
+        "node": ">=10.13.0"
+      }
+    },
+    "node_modules/globals": {
+      "version": "17.5.0",
+      "resolved": "https://registry.npmjs.org/globals/-/globals-17.5.0.tgz",
+      "integrity": "sha512-qoV+HK2yFl/366t2/Cb3+xxPUo5BuMynomoDmiaZBIdbs+0pYbjfZU+twLhGKp4uCZ/+NbtpVepH5bGCxRyy2g==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=18"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/has-flag": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
+      "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/hermes-estree": {
+      "version": "0.25.1",
+      "resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz",
+      "integrity": "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/hermes-parser": {
+      "version": "0.25.1",
+      "resolved": "https://registry.npmjs.org/hermes-parser/-/hermes-parser-0.25.1.tgz",
+      "integrity": "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "hermes-estree": "0.25.1"
+      }
+    },
+    "node_modules/ignore": {
+      "version": "5.3.2",
+      "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz",
+      "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">= 4"
+      }
+    },
+    "node_modules/import-fresh": {
+      "version": "3.3.1",
+      "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.1.tgz",
+      "integrity": "sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "parent-module": "^1.0.0",
+        "resolve-from": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/imurmurhash": {
+      "version": "0.1.4",
+      "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz",
+      "integrity": "sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=0.8.19"
+      }
+    },
+    "node_modules/is-extglob": {
+      "version": "2.1.1",
+      "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
+      "integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/is-glob": {
+      "version": "4.0.3",
+      "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
+      "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "is-extglob": "^2.1.1"
+      },
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/isexe": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
+      "integrity": "sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==",
+      "dev": true,
+      "license": "ISC"
+    },
+    "node_modules/js-tokens": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
+      "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/js-yaml": {
+      "version": "4.1.1",
+      "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.1.tgz",
+      "integrity": "sha512-qQKT4zQxXl8lLwBtHMWwaTcGfFOZviOJet3Oy/xmGk2gZH677CJM9EvtfdSkgWcATZhj/55JZ0rmy3myCT5lsA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "argparse": "^2.0.1"
+      },
+      "bin": {
+        "js-yaml": "bin/js-yaml.js"
+      }
+    },
+    "node_modules/jsesc": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-3.1.0.tgz",
+      "integrity": "sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA==",
+      "dev": true,
+      "license": "MIT",
+      "bin": {
+        "jsesc": "bin/jsesc"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/json-buffer": {
+      "version": "3.0.1",
+      "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz",
+      "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/json-schema-traverse": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz",
+      "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/json-stable-stringify-without-jsonify": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz",
+      "integrity": "sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/json5": {
+      "version": "2.2.3",
+      "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
+      "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
+      "dev": true,
+      "license": "MIT",
+      "bin": {
+        "json5": "lib/cli.js"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/keyv": {
+      "version": "4.5.4",
+      "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.5.4.tgz",
+      "integrity": "sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "json-buffer": "3.0.1"
+      }
+    },
+    "node_modules/levn": {
+      "version": "0.4.1",
+      "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz",
+      "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "prelude-ls": "^1.2.1",
+        "type-check": "~0.4.0"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/lightningcss": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss/-/lightningcss-1.32.0.tgz",
+      "integrity": "sha512-NXYBzinNrblfraPGyrbPoD19C1h9lfI/1mzgWYvXUTe414Gz/X1FD2XBZSZM7rRTrMA8JL3OtAaGifrIKhQ5yQ==",
+      "dev": true,
+      "license": "MPL-2.0",
+      "dependencies": {
+        "detect-libc": "^2.0.3"
+      },
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      },
+      "optionalDependencies": {
+        "lightningcss-android-arm64": "1.32.0",
+        "lightningcss-darwin-arm64": "1.32.0",
+        "lightningcss-darwin-x64": "1.32.0",
+        "lightningcss-freebsd-x64": "1.32.0",
+        "lightningcss-linux-arm-gnueabihf": "1.32.0",
+        "lightningcss-linux-arm64-gnu": "1.32.0",
+        "lightningcss-linux-arm64-musl": "1.32.0",
+        "lightningcss-linux-x64-gnu": "1.32.0",
+        "lightningcss-linux-x64-musl": "1.32.0",
+        "lightningcss-win32-arm64-msvc": "1.32.0",
+        "lightningcss-win32-x64-msvc": "1.32.0"
+      }
+    },
+    "node_modules/lightningcss-android-arm64": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-android-arm64/-/lightningcss-android-arm64-1.32.0.tgz",
+      "integrity": "sha512-YK7/ClTt4kAK0vo6w3X+Pnm0D2cf2vPHbhOXdoNti1Ga0al1P4TBZhwjATvjNwLEBCnKvjJc2jQgHXH0NEwlAg==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "android"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-darwin-arm64": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-darwin-arm64/-/lightningcss-darwin-arm64-1.32.0.tgz",
+      "integrity": "sha512-RzeG9Ju5bag2Bv1/lwlVJvBE3q6TtXskdZLLCyfg5pt+HLz9BqlICO7LZM7VHNTTn/5PRhHFBSjk5lc4cmscPQ==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-darwin-x64": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-darwin-x64/-/lightningcss-darwin-x64-1.32.0.tgz",
+      "integrity": "sha512-U+QsBp2m/s2wqpUYT/6wnlagdZbtZdndSmut/NJqlCcMLTWp5muCrID+K5UJ6jqD2BFshejCYXniPDbNh73V8w==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "darwin"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-freebsd-x64": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-freebsd-x64/-/lightningcss-freebsd-x64-1.32.0.tgz",
+      "integrity": "sha512-JCTigedEksZk3tHTTthnMdVfGf61Fky8Ji2E4YjUTEQX14xiy/lTzXnu1vwiZe3bYe0q+SpsSH/CTeDXK6WHig==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "freebsd"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-linux-arm-gnueabihf": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-linux-arm-gnueabihf/-/lightningcss-linux-arm-gnueabihf-1.32.0.tgz",
+      "integrity": "sha512-x6rnnpRa2GL0zQOkt6rts3YDPzduLpWvwAF6EMhXFVZXD4tPrBkEFqzGowzCsIWsPjqSK+tyNEODUBXeeVHSkw==",
+      "cpu": [
+        "arm"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-linux-arm64-gnu": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-gnu/-/lightningcss-linux-arm64-gnu-1.32.0.tgz",
+      "integrity": "sha512-0nnMyoyOLRJXfbMOilaSRcLH3Jw5z9HDNGfT/gwCPgaDjnx0i8w7vBzFLFR1f6CMLKF8gVbebmkUN3fa/kQJpQ==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "libc": [
+        "glibc"
+      ],
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-linux-arm64-musl": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-linux-arm64-musl/-/lightningcss-linux-arm64-musl-1.32.0.tgz",
+      "integrity": "sha512-UpQkoenr4UJEzgVIYpI80lDFvRmPVg6oqboNHfoH4CQIfNA+HOrZ7Mo7KZP02dC6LjghPQJeBsvXhJod/wnIBg==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "libc": [
+        "musl"
+      ],
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-linux-x64-gnu": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-gnu/-/lightningcss-linux-x64-gnu-1.32.0.tgz",
+      "integrity": "sha512-V7Qr52IhZmdKPVr+Vtw8o+WLsQJYCTd8loIfpDaMRWGUZfBOYEJeyJIkqGIDMZPwPx24pUMfwSxxI8phr/MbOA==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "libc": [
+        "glibc"
+      ],
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-linux-x64-musl": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-linux-x64-musl/-/lightningcss-linux-x64-musl-1.32.0.tgz",
+      "integrity": "sha512-bYcLp+Vb0awsiXg/80uCRezCYHNg1/l3mt0gzHnWV9XP1W5sKa5/TCdGWaR/zBM2PeF/HbsQv/j2URNOiVuxWg==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "libc": [
+        "musl"
+      ],
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "linux"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-win32-arm64-msvc": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-win32-arm64-msvc/-/lightningcss-win32-arm64-msvc-1.32.0.tgz",
+      "integrity": "sha512-8SbC8BR40pS6baCM8sbtYDSwEVQd4JlFTOlaD3gWGHfThTcABnNDBda6eTZeqbofalIJhFx0qKzgHJmcPTnGdw==",
+      "cpu": [
+        "arm64"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/lightningcss-win32-x64-msvc": {
+      "version": "1.32.0",
+      "resolved": "https://registry.npmjs.org/lightningcss-win32-x64-msvc/-/lightningcss-win32-x64-msvc-1.32.0.tgz",
+      "integrity": "sha512-Amq9B/SoZYdDi1kFrojnoqPLxYhQ4Wo5XiL8EVJrVsB8ARoC1PWW6VGtT0WKCemjy8aC+louJnjS7U18x3b06Q==",
+      "cpu": [
+        "x64"
+      ],
+      "dev": true,
+      "license": "MPL-2.0",
+      "optional": true,
+      "os": [
+        "win32"
+      ],
+      "engines": {
+        "node": ">= 12.0.0"
+      },
+      "funding": {
+        "type": "opencollective",
+        "url": "https://opencollective.com/parcel"
+      }
+    },
+    "node_modules/locate-path": {
+      "version": "6.0.0",
+      "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz",
+      "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "p-locate": "^5.0.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/lodash.merge": {
+      "version": "4.6.2",
+      "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz",
+      "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/lru-cache": {
+      "version": "5.1.1",
+      "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
+      "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==",
+      "dev": true,
+      "license": "ISC",
+      "dependencies": {
+        "yallist": "^3.0.2"
+      }
+    },
+    "node_modules/minimatch": {
+      "version": "3.1.5",
+      "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.5.tgz",
+      "integrity": "sha512-VgjWUsnnT6n+NUk6eZq77zeFdpW2LWDzP6zFGrCbHXiYNul5Dzqk2HHQ5uFH2DNW5Xbp8+jVzaeNt94ssEEl4w==",
+      "dev": true,
+      "license": "ISC",
+      "dependencies": {
+        "brace-expansion": "^1.1.7"
+      },
+      "engines": {
+        "node": "*"
+      }
+    },
+    "node_modules/ms": {
+      "version": "2.1.3",
+      "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+      "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/nanoid": {
+      "version": "3.3.11",
+      "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.11.tgz",
+      "integrity": "sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/ai"
+        }
+      ],
+      "license": "MIT",
+      "bin": {
+        "nanoid": "bin/nanoid.cjs"
+      },
+      "engines": {
+        "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
+      }
+    },
+    "node_modules/natural-compare": {
+      "version": "1.4.0",
+      "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
+      "integrity": "sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/node-releases": {
+      "version": "2.0.38",
+      "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.38.tgz",
+      "integrity": "sha512-3qT/88Y3FbH/Kx4szpQQ4HzUbVrHPKTLVpVocKiLfoYvw9XSGOX2FmD2d6DrXbVYyAQTF2HeF6My8jmzx7/CRw==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/optionator": {
+      "version": "0.9.4",
+      "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz",
+      "integrity": "sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "deep-is": "^0.1.3",
+        "fast-levenshtein": "^2.0.6",
+        "levn": "^0.4.1",
+        "prelude-ls": "^1.2.1",
+        "type-check": "^0.4.0",
+        "word-wrap": "^1.2.5"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/p-limit": {
+      "version": "3.1.0",
+      "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz",
+      "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "yocto-queue": "^0.1.0"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/p-locate": {
+      "version": "5.0.0",
+      "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz",
+      "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "p-limit": "^3.0.2"
+      },
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/parent-module": {
+      "version": "1.0.1",
+      "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz",
+      "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "callsites": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/path-exists": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz",
+      "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/path-key": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz",
+      "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/picocolors": {
+      "version": "1.1.1",
+      "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
+      "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
+      "dev": true,
+      "license": "ISC"
+    },
+    "node_modules/picomatch": {
+      "version": "4.0.4",
+      "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz",
+      "integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=12"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/jonschlinkert"
+      }
+    },
+    "node_modules/postcss": {
+      "version": "8.5.10",
+      "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.10.tgz",
+      "integrity": "sha512-pMMHxBOZKFU6HgAZ4eyGnwXF/EvPGGqUr0MnZ5+99485wwW41kW91A4LOGxSHhgugZmSChL5AlElNdwlNgcnLQ==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "opencollective",
+          "url": "https://opencollective.com/postcss/"
+        },
+        {
+          "type": "tidelift",
+          "url": "https://tidelift.com/funding/github/npm/postcss"
+        },
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/ai"
+        }
+      ],
+      "license": "MIT",
+      "dependencies": {
+        "nanoid": "^3.3.11",
+        "picocolors": "^1.1.1",
+        "source-map-js": "^1.2.1"
+      },
+      "engines": {
+        "node": "^10 || ^12 || >=14"
+      }
+    },
+    "node_modules/prelude-ls": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
+      "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/punycode": {
+      "version": "2.3.1",
+      "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
+      "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=6"
+      }
+    },
+    "node_modules/react": {
+      "version": "19.2.5",
+      "resolved": "https://registry.npmjs.org/react/-/react-19.2.5.tgz",
+      "integrity": "sha512-llUJLzz1zTUBrskt2pwZgLq59AemifIftw4aB7JxOqf1HY2FDaGDxgwpAPVzHU1kdWabH7FauP4i1oEeer2WCA==",
+      "license": "MIT",
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/react-dom": {
+      "version": "19.2.5",
+      "resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.5.tgz",
+      "integrity": "sha512-J5bAZz+DXMMwW/wV3xzKke59Af6CHY7G4uYLN1OvBcKEsWOs4pQExj86BBKamxl/Ik5bx9whOrvBlSDfWzgSag==",
+      "license": "MIT",
+      "dependencies": {
+        "scheduler": "^0.27.0"
+      },
+      "peerDependencies": {
+        "react": "^19.2.5"
+      }
+    },
+    "node_modules/resolve-from": {
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz",
+      "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=4"
+      }
+    },
+    "node_modules/rolldown": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.0-rc.16.tgz",
+      "integrity": "sha512-rzi5WqKzEZw3SooTt7cgm4eqIoujPIyGcJNGFL7iPEuajQw7vxMHUkXylu4/vhCkJGXsgRmxqMKXUpT6FEgl0g==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "@oxc-project/types": "=0.126.0",
+        "@rolldown/pluginutils": "1.0.0-rc.16"
+      },
+      "bin": {
+        "rolldown": "bin/cli.mjs"
+      },
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      },
+      "optionalDependencies": {
+        "@rolldown/binding-android-arm64": "1.0.0-rc.16",
+        "@rolldown/binding-darwin-arm64": "1.0.0-rc.16",
+        "@rolldown/binding-darwin-x64": "1.0.0-rc.16",
+        "@rolldown/binding-freebsd-x64": "1.0.0-rc.16",
+        "@rolldown/binding-linux-arm-gnueabihf": "1.0.0-rc.16",
+        "@rolldown/binding-linux-arm64-gnu": "1.0.0-rc.16",
+        "@rolldown/binding-linux-arm64-musl": "1.0.0-rc.16",
+        "@rolldown/binding-linux-ppc64-gnu": "1.0.0-rc.16",
+        "@rolldown/binding-linux-s390x-gnu": "1.0.0-rc.16",
+        "@rolldown/binding-linux-x64-gnu": "1.0.0-rc.16",
+        "@rolldown/binding-linux-x64-musl": "1.0.0-rc.16",
+        "@rolldown/binding-openharmony-arm64": "1.0.0-rc.16",
+        "@rolldown/binding-wasm32-wasi": "1.0.0-rc.16",
+        "@rolldown/binding-win32-arm64-msvc": "1.0.0-rc.16",
+        "@rolldown/binding-win32-x64-msvc": "1.0.0-rc.16"
+      }
+    },
+    "node_modules/rolldown/node_modules/@rolldown/pluginutils": {
+      "version": "1.0.0-rc.16",
+      "resolved": "https://registry.npmjs.org/@rolldown/pluginutils/-/pluginutils-1.0.0-rc.16.tgz",
+      "integrity": "sha512-45+YtqxLYKDWQouLKCrpIZhke+nXxhsw+qAHVzHDVwttyBlHNBVs2K25rDXrZzhpTp9w1FlAlvweV1H++fdZoA==",
+      "dev": true,
+      "license": "MIT"
+    },
+    "node_modules/scheduler": {
+      "version": "0.27.0",
+      "resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
+      "integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
+      "license": "MIT"
+    },
+    "node_modules/semver": {
+      "version": "6.3.1",
+      "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
+      "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==",
+      "dev": true,
+      "license": "ISC",
+      "bin": {
+        "semver": "bin/semver.js"
+      }
+    },
+    "node_modules/shebang-command": {
+      "version": "2.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
+      "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "shebang-regex": "^3.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/shebang-regex": {
+      "version": "3.0.0",
+      "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz",
+      "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/source-map-js": {
+      "version": "1.2.1",
+      "resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
+      "integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
+      "dev": true,
+      "license": "BSD-3-Clause",
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/strip-json-comments": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-3.1.1.tgz",
+      "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=8"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/supports-color": {
+      "version": "7.2.0",
+      "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
+      "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "has-flag": "^4.0.0"
+      },
+      "engines": {
+        "node": ">=8"
+      }
+    },
+    "node_modules/tinyglobby": {
+      "version": "0.2.16",
+      "resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz",
+      "integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "fdir": "^6.5.0",
+        "picomatch": "^4.0.4"
+      },
+      "engines": {
+        "node": ">=12.0.0"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/SuperchupuDev"
+      }
+    },
+    "node_modules/tslib": {
+      "version": "2.8.1",
+      "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
+      "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==",
+      "dev": true,
+      "license": "0BSD",
+      "optional": true
+    },
+    "node_modules/type-check": {
+      "version": "0.4.0",
+      "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz",
+      "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "prelude-ls": "^1.2.1"
+      },
+      "engines": {
+        "node": ">= 0.8.0"
+      }
+    },
+    "node_modules/update-browserslist-db": {
+      "version": "1.2.3",
+      "resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz",
+      "integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==",
+      "dev": true,
+      "funding": [
+        {
+          "type": "opencollective",
+          "url": "https://opencollective.com/browserslist"
+        },
+        {
+          "type": "tidelift",
+          "url": "https://tidelift.com/funding/github/npm/browserslist"
+        },
+        {
+          "type": "github",
+          "url": "https://github.com/sponsors/ai"
+        }
+      ],
+      "license": "MIT",
+      "dependencies": {
+        "escalade": "^3.2.0",
+        "picocolors": "^1.1.1"
+      },
+      "bin": {
+        "update-browserslist-db": "cli.js"
+      },
+      "peerDependencies": {
+        "browserslist": ">= 4.21.0"
+      }
+    },
+    "node_modules/uri-js": {
+      "version": "4.4.1",
+      "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz",
+      "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==",
+      "dev": true,
+      "license": "BSD-2-Clause",
+      "dependencies": {
+        "punycode": "^2.1.0"
+      }
+    },
+    "node_modules/vite": {
+      "version": "8.0.9",
+      "resolved": "https://registry.npmjs.org/vite/-/vite-8.0.9.tgz",
+      "integrity": "sha512-t7g7GVRpMXjNpa67HaVWI/8BWtdVIQPCL2WoozXXA7LBGEFK4AkkKkHx2hAQf5x1GZSlcmEDPkVLSGahxnEEZw==",
+      "dev": true,
+      "license": "MIT",
+      "dependencies": {
+        "lightningcss": "^1.32.0",
+        "picomatch": "^4.0.4",
+        "postcss": "^8.5.10",
+        "rolldown": "1.0.0-rc.16",
+        "tinyglobby": "^0.2.16"
+      },
+      "bin": {
+        "vite": "bin/vite.js"
+      },
+      "engines": {
+        "node": "^20.19.0 || >=22.12.0"
+      },
+      "funding": {
+        "url": "https://github.com/vitejs/vite?sponsor=1"
+      },
+      "optionalDependencies": {
+        "fsevents": "~2.3.3"
+      },
+      "peerDependencies": {
+        "@types/node": "^20.19.0 || >=22.12.0",
+        "@vitejs/devtools": "^0.1.0",
+        "esbuild": "^0.27.0 || ^0.28.0",
+        "jiti": ">=1.21.0",
+        "less": "^4.0.0",
+        "sass": "^1.70.0",
+        "sass-embedded": "^1.70.0",
+        "stylus": ">=0.54.8",
+        "sugarss": "^5.0.0",
+        "terser": "^5.16.0",
+        "tsx": "^4.8.1",
+        "yaml": "^2.4.2"
+      },
+      "peerDependenciesMeta": {
+        "@types/node": {
+          "optional": true
+        },
+        "@vitejs/devtools": {
+          "optional": true
+        },
+        "esbuild": {
+          "optional": true
+        },
+        "jiti": {
+          "optional": true
+        },
+        "less": {
+          "optional": true
+        },
+        "sass": {
+          "optional": true
+        },
+        "sass-embedded": {
+          "optional": true
+        },
+        "stylus": {
+          "optional": true
+        },
+        "sugarss": {
+          "optional": true
+        },
+        "terser": {
+          "optional": true
+        },
+        "tsx": {
+          "optional": true
+        },
+        "yaml": {
+          "optional": true
+        }
+      }
+    },
+    "node_modules/which": {
+      "version": "2.0.2",
+      "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
+      "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==",
+      "dev": true,
+      "license": "ISC",
+      "dependencies": {
+        "isexe": "^2.0.0"
+      },
+      "bin": {
+        "node-which": "bin/node-which"
+      },
+      "engines": {
+        "node": ">= 8"
+      }
+    },
+    "node_modules/word-wrap": {
+      "version": "1.2.5",
+      "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.5.tgz",
+      "integrity": "sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=0.10.0"
+      }
+    },
+    "node_modules/yallist": {
+      "version": "3.1.1",
+      "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz",
+      "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==",
+      "dev": true,
+      "license": "ISC"
+    },
+    "node_modules/yocto-queue": {
+      "version": "0.1.0",
+      "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz",
+      "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=10"
+      },
+      "funding": {
+        "url": "https://github.com/sponsors/sindresorhus"
+      }
+    },
+    "node_modules/zod": {
+      "version": "4.3.6",
+      "resolved": "https://registry.npmjs.org/zod/-/zod-4.3.6.tgz",
+      "integrity": "sha512-rftlrkhHZOcjDwkGlnUtZZkvaPHCsDATp4pGpuOOMDaTdDDXF91wuVDJoWoPsKX/3YPQ5fHuF3STjcYyKr+Qhg==",
+      "dev": true,
+      "license": "MIT",
+      "funding": {
+        "url": "https://github.com/sponsors/colinhacks"
+      }
+    },
+    "node_modules/zod-validation-error": {
+      "version": "4.0.2",
+      "resolved": "https://registry.npmjs.org/zod-validation-error/-/zod-validation-error-4.0.2.tgz",
+      "integrity": "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ==",
+      "dev": true,
+      "license": "MIT",
+      "engines": {
+        "node": ">=18.0.0"
+      },
+      "peerDependencies": {
+        "zod": "^3.25.0 || ^4.0.0"
+      }
+    }
+  }
+}

+ 27 - 0
04 - Scripts/sfcr-app/package.json

@@ -0,0 +1,27 @@
+{
+  "name": "sfcr-app",
+  "private": true,
+  "version": "0.0.0",
+  "type": "module",
+  "scripts": {
+    "dev": "vite",
+    "build": "vite build",
+    "lint": "eslint .",
+    "preview": "vite preview"
+  },
+  "dependencies": {
+    "react": "^19.2.5",
+    "react-dom": "^19.2.5"
+  },
+  "devDependencies": {
+    "@eslint/js": "^9.39.4",
+    "@types/react": "^19.2.14",
+    "@types/react-dom": "^19.2.3",
+    "@vitejs/plugin-react": "^6.0.1",
+    "eslint": "^9.39.4",
+    "eslint-plugin-react-hooks": "^7.1.1",
+    "eslint-plugin-react-refresh": "^0.5.2",
+    "globals": "^17.5.0",
+    "vite": "^8.0.9"
+  }
+}

File diff suppressed because it is too large
+ 0 - 0
04 - Scripts/sfcr-app/public/favicon.svg


+ 24 - 0
04 - Scripts/sfcr-app/public/icons.svg

@@ -0,0 +1,24 @@
+<svg xmlns="http://www.w3.org/2000/svg">
+  <symbol id="bluesky-icon" viewBox="0 0 16 17">
+    <g clip-path="url(#bluesky-clip)"><path fill="#08060d" d="M7.75 7.735c-.693-1.348-2.58-3.86-4.334-5.097-1.68-1.187-2.32-.981-2.74-.79C.188 2.065.1 2.812.1 3.251s.241 3.602.398 4.13c.52 1.744 2.367 2.333 4.07 2.145-2.495.37-4.71 1.278-1.805 4.512 3.196 3.309 4.38-.71 4.987-2.746.608 2.036 1.307 5.91 4.93 2.746 2.72-2.746.747-4.143-1.747-4.512 1.702.189 3.55-.4 4.07-2.145.156-.528.397-3.691.397-4.13s-.088-1.186-.575-1.406c-.42-.19-1.06-.395-2.741.79-1.755 1.24-3.64 3.752-4.334 5.099"/></g>
+    <defs><clipPath id="bluesky-clip"><path fill="#fff" d="M.1.85h15.3v15.3H.1z"/></clipPath></defs>
+  </symbol>
+  <symbol id="discord-icon" viewBox="0 0 20 19">
+    <path fill="#08060d" d="M16.224 3.768a14.5 14.5 0 0 0-3.67-1.153c-.158.286-.343.67-.47.976a13.5 13.5 0 0 0-4.067 0c-.128-.306-.317-.69-.476-.976A14.4 14.4 0 0 0 3.868 3.77C1.546 7.28.916 10.703 1.231 14.077a14.7 14.7 0 0 0 4.5 2.306q.545-.748.965-1.587a9.5 9.5 0 0 1-1.518-.74q.191-.14.372-.293c2.927 1.369 6.107 1.369 8.999 0q.183.152.372.294-.723.437-1.52.74.418.838.963 1.588a14.6 14.6 0 0 0 4.504-2.308c.37-3.911-.63-7.302-2.644-10.309m-9.13 8.234c-.878 0-1.599-.82-1.599-1.82 0-.998.705-1.82 1.6-1.82.894 0 1.614.82 1.599 1.82.001 1-.705 1.82-1.6 1.82m5.91 0c-.878 0-1.599-.82-1.599-1.82 0-.998.705-1.82 1.6-1.82.893 0 1.614.82 1.599 1.82 0 1-.706 1.82-1.6 1.82"/>
+  </symbol>
+  <symbol id="documentation-icon" viewBox="0 0 21 20">
+    <path fill="none" stroke="#aa3bff" stroke-linecap="round" stroke-linejoin="round" stroke-width="1.35" d="m15.5 13.333 1.533 1.322c.645.555.967.833.967 1.178s-.322.623-.967 1.179L15.5 18.333m-3.333-5-1.534 1.322c-.644.555-.966.833-.966 1.178s.322.623.966 1.179l1.534 1.321"/>
+    <path fill="none" stroke="#aa3bff" stroke-linecap="round" stroke-linejoin="round" stroke-width="1.35" d="M17.167 10.836v-4.32c0-1.41 0-2.117-.224-2.68-.359-.906-1.118-1.621-2.08-1.96-.599-.21-1.349-.21-2.848-.21-2.623 0-3.935 0-4.983.369-1.684.591-3.013 1.842-3.641 3.428C3 6.449 3 7.684 3 10.154v2.122c0 2.558 0 3.838.706 4.726q.306.383.713.671c.76.536 1.79.64 3.581.66"/>
+    <path fill="none" stroke="#aa3bff" stroke-linecap="round" stroke-linejoin="round" stroke-width="1.35" d="M3 10a2.78 2.78 0 0 1 2.778-2.778c.555 0 1.209.097 1.748-.047.48-.129.854-.503.982-.982.145-.54.048-1.194.048-1.749a2.78 2.78 0 0 1 2.777-2.777"/>
+  </symbol>
+  <symbol id="github-icon" viewBox="0 0 19 19">
+    <path fill="#08060d" fill-rule="evenodd" d="M9.356 1.85C5.05 1.85 1.57 5.356 1.57 9.694a7.84 7.84 0 0 0 5.324 7.44c.387.079.528-.168.528-.376 0-.182-.013-.805-.013-1.454-2.165.467-2.616-.935-2.616-.935-.349-.91-.864-1.143-.864-1.143-.71-.48.051-.48.051-.48.787.051 1.2.805 1.2.805.695 1.194 1.817.857 2.268.649.064-.507.27-.857.49-1.052-1.728-.182-3.545-.857-3.545-3.87 0-.857.31-1.558.8-2.104-.078-.195-.349-1 .077-2.078 0 0 .657-.208 2.14.805a7.5 7.5 0 0 1 1.946-.26c.657 0 1.328.092 1.946.26 1.483-1.013 2.14-.805 2.14-.805.426 1.078.155 1.883.078 2.078.502.546.799 1.247.799 2.104 0 3.013-1.818 3.675-3.558 3.87.284.247.528.714.528 1.454 0 1.052-.012 1.896-.012 2.156 0 .208.142.455.528.377a7.84 7.84 0 0 0 5.324-7.441c.013-4.338-3.48-7.844-7.773-7.844" clip-rule="evenodd"/>
+  </symbol>
+  <symbol id="social-icon" viewBox="0 0 20 20">
+    <path fill="none" stroke="#aa3bff" stroke-linecap="round" stroke-linejoin="round" stroke-width="1.35" d="M12.5 6.667a4.167 4.167 0 1 0-8.334 0 4.167 4.167 0 0 0 8.334 0"/>
+    <path fill="none" stroke="#aa3bff" stroke-linecap="round" stroke-linejoin="round" stroke-width="1.35" d="M2.5 16.667a5.833 5.833 0 0 1 8.75-5.053m3.837.474.513 1.035c.07.144.257.282.414.309l.93.155c.596.1.736.536.307.965l-.723.73a.64.64 0 0 0-.152.531l.207.903c.164.715-.213.991-.84.618l-.872-.52a.63.63 0 0 0-.577 0l-.872.52c-.624.373-1.003.094-.84-.618l.207-.903a.64.64 0 0 0-.152-.532l-.723-.729c-.426-.43-.289-.864.306-.964l.93-.156a.64.64 0 0 0 .412-.31l.513-1.034c.28-.562.735-.562 1.012 0"/>
+  </symbol>
+  <symbol id="x-icon" viewBox="0 0 19 19">
+    <path fill="#08060d" fill-rule="evenodd" d="M1.893 1.98c.052.072 1.245 1.769 2.653 3.77l2.892 4.114c.183.261.333.48.333.486s-.068.089-.152.183l-.522.593-.765.867-3.597 4.087c-.375.426-.734.834-.798.905a1 1 0 0 0-.118.148c0 .01.236.017.664.017h.663l.729-.83c.4-.457.796-.906.879-.999a692 692 0 0 0 1.794-2.038c.034-.037.301-.34.594-.675l.551-.624.345-.392a7 7 0 0 1 .34-.374c.006 0 .93 1.306 2.052 2.903l2.084 2.965.045.063h2.275c1.87 0 2.273-.003 2.266-.021-.008-.02-1.098-1.572-3.894-5.547-2.013-2.862-2.28-3.246-2.273-3.266.008-.019.282-.332 2.085-2.38l2-2.274 1.567-1.782c.022-.028-.016-.03-.65-.03h-.674l-.3.342a871 871 0 0 1-1.782 2.025c-.067.075-.405.458-.75.852a100 100 0 0 1-.803.91c-.148.172-.299.344-.99 1.127-.304.343-.32.358-.345.327-.015-.019-.904-1.282-1.976-2.808L6.365 1.85H1.8zm1.782.91 8.078 11.294c.772 1.08 1.413 1.973 1.425 1.984.016.017.241.02 1.05.017l1.03-.004-2.694-3.766L7.796 5.75 5.722 2.852l-1.039-.004-1.039-.004z" clip-rule="evenodd"/>
+  </symbol>
+</svg>

+ 97 - 0
04 - Scripts/sfcr-app/src/App.css

@@ -0,0 +1,97 @@
+@import url('https://fonts.googleapis.com/css2?family=DM+Sans:wght@400;500;600;700;800&display=swap');
+
+:root {
+  --bg: #0d0f14;
+  --surface: #141720;
+  --surface-hover: #1c2030;
+  --border: #252a38;
+  --border-light: #2e3448;
+  --accent: #3b82f6;
+  --accent-glow: rgba(59, 130, 246, 0.2);
+  --accent-soft: rgba(59, 130, 246, 0.08);
+  --gold: #f59e0b;
+  --gold-soft: rgba(245, 158, 11, 0.12);
+  --text: #e8ecf4;
+  --text-muted: #6b7594;
+  --text-dim: #9aa3bc;
+  --green: #10b981;
+  --green-soft: rgba(16, 185, 129, 0.1);
+}
+
+* { box-sizing: border-box; }
+body { margin: 0; background: var(--bg); color: var(--text); font-family: 'DM Sans', sans-serif; }
+
+/* Scrollbar */
+::-webkit-scrollbar { width: 6px; }
+::-webkit-scrollbar-thumb { background: var(--border); border-radius: 10px; }
+
+.app-root { display: flex; height: 100vh; overflow: hidden; }
+
+/* Sidebar */
+.sidebar {
+  background: var(--surface);
+  border-right: 1px solid var(--border);
+  display: flex;
+  flex-direction: column;
+  transition: width 0.25s ease;
+}
+
+.nav-btn {
+  display: flex; align-items: center; gap: 12px; padding: 12px;
+  background: none; border: none; color: var(--text-muted);
+  cursor: pointer; width: 100%; border-radius: 8px; transition: 0.2s;
+}
+
+.nav-btn.active { background: var(--accent-soft); color: var(--accent); }
+
+/* Main Panels */
+.main-content { flex: 1; overflow-y: auto; padding: 32px; }
+
+.page-header { display: flex; justify-content: space-between; margin-bottom: 30px; }
+
+.card {
+  background: var(--surface);
+  border: 1px solid var(--border);
+  border-radius: 12px;
+  padding: 20px;
+  margin-bottom: 20px;
+}
+
+.extract-btn {
+  background: var(--accent); color: white; border: none;
+  padding: 12px 24px; border-radius: 8px; font-weight: 700;
+  cursor: pointer; transition: 0.2s; box-shadow: 0 4px 15px var(--accent-glow);
+}
+
+.extract-btn:hover { transform: translateY(-2px); opacity: 0.9; }
+
+/* Grid Layout */
+.two-col { display: grid; grid-template-columns: 1fr 1fr; gap: 24px; }
+
+/* Company Row */
+.company-row {
+  display: flex; align-items: center; gap: 12px; padding: 10px;
+  border-radius: 8px; cursor: pointer; transition: 0.15s;
+}
+.company-row:hover { background: var(--surface-hover); }
+.company-row.selected { background: var(--accent-soft); border: 1px solid var(--accent-soft); }
+
+.avatar {
+  width: 32px; height: 32px; border-radius: 6px;
+  background: linear-gradient(135deg, #1e3a5f, #2d5a8e);
+  display: flex; align-items: center; justify-content: center;
+  font-size: 11px; font-weight: bold; color: #7fb3e8;
+}
+
+/* Utils */
+.chip-grid { display: flex; gap: 8px; flex-wrap: wrap; }
+.chip {
+  padding: 8px 16px; border-radius: 6px; border: 1px solid var(--border);
+  background: var(--bg); cursor: pointer; color: var(--text-dim);
+}
+.chip.active { border-color: var(--accent); color: var(--accent); background: var(--accent-soft); }
+
+.prompt-area {
+  width: 100%; background: var(--bg); border: 1px solid var(--border);
+  border-radius: 8px; color: var(--text); padding: 12px; resize: none;
+}

+ 457 - 0
04 - Scripts/sfcr-app/src/App.jsx

@@ -0,0 +1,457 @@
+import { useState, useMemo } from "react";
+import { ALL_COMPANIES, SFCR_SECTIONS, YEARS, CURRENT_YEAR, sectionLabels } from "./constants";
+import { styles, cssString } from "./styles";
+import { C } from "./styles"; // Import des couleurs pour usage direct si besoin
+
+// --- COMPOSANT DE SUIVI (Externe à App pour la performance) ---
+const ExtractionStatusView = ({ selectedCompanies, globalSections, onClose }) => {
+  return (
+    <div style={styles.extractionWrapper}>
+      <div style={styles.extractionHeader}>
+        <div>
+          <h2 style={styles.pageTitle}>Analyse Multi-Agents</h2>
+          <p style={styles.pageSubtitle}>Traitement des rapports Solvabilité II en cours...</p>
+        </div>
+        <button onClick={onClose} style={styles.showMoreBtn}>Interrompre l'analyse</button>
+      </div>
+
+      <div style={styles.extractionList}>
+        {selectedCompanies.map((company) => (
+          <div key={company.id} style={styles.extractionCard}>
+            <div style={styles.companyStatusHeader}>
+              <div style={styles.companyAvatar}>{company.logo}</div>
+              <div style={{ flex: 1 }}>
+                <h4 style={{ margin: 0, fontSize: '14px', color: '#e8ecf4' }}>{company.name}</h4>
+                <span style={{ fontSize: '11px', color: C.accent }}>
+                  Extraction en cours... <span className="loader-dots"></span>
+                </span>
+              </div>
+            </div>
+
+            <div style={styles.stepsTimeline}>
+              <div style={styles.stepItemActive}>
+                <span style={styles.stepIcon}>📄</span>
+                <span>Fichier PDF chargé</span>
+              </div>
+              
+              {globalSections.length > 0 ? (
+                globalSections.map(section => (
+                  <div key={section} style={styles.stepItemPending}>
+                    <span style={styles.stepIcon}>⚙️</span>
+                    <span>Analyse {section}</span>
+                  </div>
+                ))
+              ) : (
+                <div style={styles.stepItemPending}>
+                  <span style={styles.stepIcon}>⚙️</span>
+                  <span>Analyse automatique</span>
+                </div>
+              )}
+
+              <div style={styles.stepItemPending}>
+                <span style={styles.stepIcon}>✅</span>
+                <span>Finalisation</span>
+              </div>
+            </div>
+          </div>
+        ))}
+      </div>
+    </div>
+  );
+};
+
+// --- COMPOSANT PRINCIPAL ---
+export default function App() {
+  const [activeNav, setActiveNav] = useState("extraction");
+  const [search, setSearch] = useState("");
+  const [showAll, setShowAll] = useState(false);
+  const [selectedCompanies, setSelectedCompanies] = useState([]);
+  const [selectedSections, setSelectedSections] = useState({});
+  const [selectedYears, setSelectedYears] = useState({});
+  const [globalSections, setGlobalSections] = useState([]);
+  const [globalYear, setGlobalYear] = useState(CURRENT_YEAR);
+  const [sectionMode, setSectionMode] = useState("global");
+  const [yearMode, setYearMode] = useState("global");
+  const [prompt, setPrompt] = useState("");
+  const [sidebarCollapsed, setSidebarCollapsed] = useState(false);
+  
+  // État pour basculer l'interface
+  const [isExtracting, setIsExtracting] = useState(false);
+
+  const filtered = useMemo(() => {
+    const base = ALL_COMPANIES.filter((c) =>
+      c.name.toLowerCase().includes(search.toLowerCase())
+    );
+    return showAll ? base : base.slice(0, 5);
+  }, [search, showAll]);
+
+  const toggleCompany = (id) => {
+    setSelectedCompanies((prev) =>
+      prev.includes(id) ? prev.filter((c) => c !== id) : [...prev, id]
+    );
+  };
+
+  const toggleAll = () => {
+    if (selectedCompanies.length === filtered.length) {
+      setSelectedCompanies([]);
+    } else {
+      setSelectedCompanies(filtered.map((c) => c.id));
+    }
+  };
+
+  const toggleGlobalSection = (s) => {
+    setGlobalSections((prev) =>
+      prev.includes(s) ? prev.filter((x) => x !== s) : [...prev, s]
+    );
+  };
+
+  const toggleCompanySection = (companyId, s) => {
+    setSelectedSections((prev) => {
+      const cur = prev[companyId] || [];
+      return {
+        ...prev,
+        [companyId]: cur.includes(s) ? cur.filter((x) => x !== s) : [...cur, s],
+      };
+    });
+  };
+
+  const setCompanyYear = (companyId, year) => {
+    setSelectedYears((prev) => ({ ...prev, [companyId]: year }));
+  };
+
+  const selectedCompanyObjects = useMemo(() => 
+    ALL_COMPANIES.filter((c) => selectedCompanies.includes(c.id)),
+    [selectedCompanies]
+  );
+
+  const handleExtraction = async () => {
+    if (selectedCompanies.length === 0) {
+      alert("Veuillez sélectionner au moins une entreprise.");
+      return;
+    }
+
+    // 1. Afficher immédiatement l'interface d'extraction
+    setIsExtracting(true);
+
+    const inputs = {};
+    selectedCompanyObjects.forEach((company) => {
+      const sections = sectionMode === "global"
+        ? globalSections
+        : (selectedSections[company.id] || []);
+
+      const year = yearMode === "global"
+        ? globalYear
+        : (selectedYears[company.id] || CURRENT_YEAR);
+
+      if (sections.length > 0) {
+        const filename = `${company.name}_${year}.pdf`;
+        inputs[filename] = {};
+        sections.forEach((s) => {
+          inputs[filename][s] = [];
+        });
+      }
+    });
+
+    try {
+      const response = await fetch("http://localhost:5000/run-extraction", {
+        method: "POST",
+        headers: { "Content-Type": "application/json" },
+        body: JSON.stringify({ inputs }),
+      });
+
+      const result = await response.json();
+      console.log("Résultat:", result);
+      
+      // On garde l'interface affichée mais on prévient que c'est fini
+      alert("Extraction terminée avec succès !");
+      // setIsExtracting(false); // Décommenter si tu veux revenir au menu automatiquement
+    } catch (err) {
+      setIsExtracting(false);
+      alert("Erreur : impossible de contacter le serveur Python.");
+    }
+  };
+
+  return (
+    <div style={styles.root}>
+      <style>{cssString}</style>
+
+      {/* Sidebar - Toujours présente */}
+      <aside style={{ ...styles.sidebar, width: sidebarCollapsed ? 64 : 220 }}>
+        <div style={styles.sidebarHeader}>
+          {!sidebarCollapsed && (
+            <span style={styles.brandText}>
+              <span style={styles.brandAccent}>SFCR</span>
+              <span style={styles.brandSub}>·extract</span>
+            </span>
+          )}
+          <button onClick={() => setSidebarCollapsed((v) => !v)} style={styles.collapseBtn}>
+            {sidebarCollapsed ? "›" : "‹"}
+          </button>
+        </div>
+
+        <nav style={styles.nav}>
+          {[
+            { id: "search", icon: "⊕", label: "Search Sources" },
+            { id: "extraction", icon: "⊞", label: "Extraction" },
+          ].map((item) => (
+            <button
+              key={item.id}
+              onClick={() => {
+                setActiveNav(item.id);
+                if (item.id === "search") setIsExtracting(false); // Reset si on change d'onglet
+              }}
+              style={{
+                ...styles.navBtn,
+                ...(activeNav === item.id ? styles.navBtnActive : {}),
+              }}
+              className="nav-btn"
+            >
+              <span style={styles.navIcon}>{item.icon}</span>
+              {!sidebarCollapsed && <span style={styles.navLabel}>{item.label}</span>}
+            </button>
+          ))}
+        </nav>
+      </aside>
+
+      {/* Main Content Area */}
+      <main style={styles.main}>
+        {activeNav === "search" ? (
+          <div style={styles.placeholderPanel}>
+            <div style={styles.placeholderIcon}>⊕</div>
+            <h2 style={styles.placeholderTitle}>Search Sources</h2>
+            <p style={styles.placeholderDesc}>Section dédiée à la recherche de sources SFCR.</p>
+          </div>
+        ) : isExtracting ? (
+          // --- VUE SUIVI (Affiche les agents en action) ---
+          <ExtractionStatusView 
+            selectedCompanies={selectedCompanyObjects}
+            globalSections={globalSections}
+            onClose={() => setIsExtracting(false)}
+          />
+        ) : (
+          // --- VUE CONFIGURATION (Interface normale) ---
+          <div style={styles.extractionPanel}>
+            <div style={styles.pageHeader}>
+              <div>
+                <h1 style={styles.pageTitle}>Extraction SFCR</h1>
+                <p style={styles.pageSubtitle}>Sélectionnez les entreprises et paramétrez l'extraction</p>
+              </div>
+              <button style={styles.extractBtn} className="extract-btn" onClick={handleExtraction}>
+                <span>▶</span> Lancer l'extraction
+              </button>
+            </div>
+
+            <div style={styles.twoCol}>
+              <div style={styles.leftCol}>
+                {/* 01. SELECTION ENTREPRISES */}
+               <section style={styles.card}>
+                  <div style={styles.cardHeader}>
+                    <span style={styles.cardBadge}>01</span>
+                    <h2 style={styles.cardTitle}>Sélection des entreprises</h2>
+                    {selectedCompanies.length > 0 && (
+                      <span style={styles.countPill}>
+                        {selectedCompanies.length} sélectionné
+                        {selectedCompanies.length > 1 ? "s" : ""}
+                      </span>
+                    )}
+                  </div>
+
+                  <div style={styles.searchRow}>
+                    <span style={styles.searchIcon}>🔍</span>
+                    <input
+                      style={styles.searchInput}
+                      placeholder="Rechercher une entreprise…"
+                      value={search}
+                      onChange={(e) => setSearch(e.target.value)}
+                    />
+                  </div>
+
+                  <div style={styles.selectAllRow}>
+                    <label style={styles.checkLabel} className="check-label">
+                      <input
+                        type="checkbox"
+                        checked={
+                          filtered.length > 0 &&
+                          filtered.every((c) => selectedCompanies.includes(c.id))
+                        }
+                        onChange={toggleAll}
+                        style={styles.hiddenCheck}
+                      />
+                      <span
+                        style={{
+                          ...styles.customCheck,
+                          ...(filtered.length > 0 &&
+                          filtered.every((c) => selectedCompanies.includes(c.id))
+                            ? styles.customCheckChecked
+                            : {}),
+                        }}
+                      >
+                        {filtered.length > 0 &&
+                          filtered.every((c) => selectedCompanies.includes(c.id)) &&
+                          "✓"}
+                      </span>
+                      <span style={styles.checkLabelText}>Tout sélectionner</span>
+                    </label>
+                  </div>
+
+                  {/* --- ZONE DE SCROLL AJOUTÉE ICI --- */}
+                  <div 
+                    style={{ 
+                      maxHeight: "300px", // Tu peux ajuster la hauteur (ex: 250px ou 400px)
+                      overflowY: "auto", 
+                      paddingRight: "5px" // Petit espace pour ne pas coller à la barre de scroll
+                    }}
+                  >
+                    <div style={styles.companyList}>
+                      {filtered.map((company) => {
+                        const checked = selectedCompanies.includes(company.id);
+                        return (
+                          <label
+                            key={company.id}
+                            style={{
+                              ...styles.companyRow,
+                              ...(checked ? styles.companyRowChecked : {}),
+                            }}
+                            className="company-row"
+                          >
+                            <input
+                              type="checkbox"
+                              checked={checked}
+                              onChange={() => toggleCompany(company.id)}
+                              style={styles.hiddenCheck}
+                            />
+                            <span style={styles.companyAvatar}>{company.logo}</span>
+                            <span style={styles.companyName}>{company.name}</span>
+                            <span
+                              style={{
+                                ...styles.customCheck,
+                                ...(checked ? styles.customCheckChecked : {}),
+                              }}
+                            >
+                              {checked && "✓"}
+                            </span>
+                          </label>
+                        );
+                      })}
+                    </div>
+                  </div>
+                  {/* --- FIN DE LA ZONE DE SCROLL --- */}
+
+                  {!showAll && ALL_COMPANIES.length > 10 && (
+                    <button
+                      style={styles.showMoreBtn}
+                      onClick={() => setShowAll(true)}
+                    >
+                      Afficher plus ({ALL_COMPANIES.length - 10} entreprises)
+                    </button>
+                  )}
+                </section>
+
+                {/* 04. ANNEE */}
+                {selectedCompanies.length > 0 && (
+                  <section style={styles.card}>
+                    <div style={styles.cardHeader}>
+                      <span style={styles.cardBadge}>04</span>
+                      <h2 style={styles.cardTitle}>Année de rapport</h2>
+                    </div>
+                    <div style={styles.modeToggle}>
+                      <button style={{...styles.modeBtn, ...(yearMode === "global" ? styles.modeBtnActive : {})}} onClick={() => setYearMode("global")}>Toutes</button>
+                      <button style={{...styles.modeBtn, ...(yearMode === "individual" ? styles.modeBtnActive : {})}} onClick={() => setYearMode("individual")}>Par entreprise</button>
+                    </div>
+                    {yearMode === "global" ? (
+                      <div style={styles.yearGrid}>
+                        {YEARS.map(y => (
+                          <button key={y} style={{...styles.yearChip, ...(globalYear === y ? styles.yearChipActive : {})}} onClick={() => setGlobalYear(y)}>{y}</button>
+                        ))}
+                      </div>
+                    ) : (
+                      <div style={styles.perCompanyList}>
+                        {selectedCompanyObjects.map(c => (
+                          <div key={c.id} style={styles.perCompanyRow}>
+                            <span style={styles.perCompanyName}>{c.name}</span>
+                            <div style={styles.miniYearRow}>
+                              {YEARS.map(y => (
+                                <button key={y} style={{...styles.miniYearChip, ...((selectedYears[c.id] || CURRENT_YEAR) === y ? styles.miniYearChipActive : {})}} onClick={() => setCompanyYear(c.id, y)}>{y}</button>
+                              ))}
+                            </div>
+                          </div>
+                        ))}
+                      </div>
+                    )}
+                  </section>
+                )}
+              </div>
+
+              <div style={styles.rightCol}>
+                {/* 03. SECTIONS */}
+                {selectedCompanies.length > 0 && (
+                  <section style={styles.card}>
+                    <div style={styles.cardHeader}>
+                      <span style={styles.cardBadge}>03</span>
+                      <h2 style={styles.cardTitle}>Sections SFCR</h2>
+                    </div>
+                    <div style={styles.modeToggle}>
+                      <button style={{...styles.modeBtn, ...(sectionMode === "global" ? styles.modeBtnActive : {})}} onClick={() => setSectionMode("global")}>Toutes</button>
+                      <button style={{...styles.modeBtn, ...(sectionMode === "individual" ? styles.modeBtnActive : {})}} onClick={() => setSectionMode("individual")}>Par entreprise</button>
+                    </div>
+                    {sectionMode === "global" ? (
+                      <div style={styles.sectionGrid}>
+                        {SFCR_SECTIONS.map(s => (
+                          <button key={s} onClick={() => toggleGlobalSection(s)} style={{...styles.sectionChip, ...(globalSections.includes(s) ? styles.sectionChipActive : {})}}>
+                            <span style={styles.sectionCode}>{s}</span>
+                            <span style={styles.sectionLabel}>{sectionLabels[s]}</span>
+                          </button>
+                        ))}
+                      </div>
+                    ) : (
+                      <div style={styles.perCompanyList}>
+                        {selectedCompanyObjects.map(c => (
+                          <div key={c.id} style={styles.perCompanyBlock}>
+                            <div style={styles.perCompanyHeader}><span style={styles.perCompanyName}>{c.name}</span></div>
+                            <div style={styles.miniSectionGrid}>
+                              {SFCR_SECTIONS.map(s => (
+                                <button key={s} onClick={() => toggleCompanySection(c.id, s)} style={{...styles.miniSectionChip, ...(selectedSections[c.id]?.includes(s) ? styles.miniSectionChipActive : {})}}>{s}</button>
+                              ))}
+                            </div>
+                          </div>
+                        ))}
+                      </div>
+                    )}
+                  </section>
+                )}
+
+                {/* 02. PROMPT */}
+                <section style={styles.card}>
+                  <div style={styles.cardHeader}>
+                    <span style={styles.cardBadge}>02</span>
+                    <h2 style={styles.cardTitle}>Prompt d'extraction</h2>
+                  </div>
+                  <div style={styles.promptWrapper}>
+                    <textarea style={styles.promptInput} placeholder="Instructions spécifiques..." value={prompt} onChange={(e) => setPrompt(e.target.value)} rows={4} />
+                  </div>
+                </section>
+
+                {/* RECAPITULATIF */}
+                {selectedCompanies.length > 0 && (
+                  <section style={styles.summaryCard}>
+                    <h3 style={styles.summaryTitle}>Récapitulatif</h3>
+                    <div style={styles.summaryGrid}>
+                      <div style={styles.summaryItem}>
+                        <span style={styles.summaryNum}>{selectedCompanies.length}</span>
+                        <span style={styles.summaryItemLabel}>Entreprise(s)</span>
+                      </div>
+                      <div style={styles.summaryItem}>
+                        <span style={styles.summaryNum}>{sectionMode === "global" ? globalSections.length : "Varié"}</span>
+                        <span style={styles.summaryItemLabel}>Section(s)</span>
+                      </div>
+                    </div>
+                  </section>
+                )}
+              </div>
+            </div>
+          </div>
+        )}
+      </main>
+    </div>
+  );
+}

BIN
04 - Scripts/sfcr-app/src/assets/hero.png


File diff suppressed because it is too large
+ 0 - 0
04 - Scripts/sfcr-app/src/assets/react.svg


File diff suppressed because it is too large
+ 0 - 0
04 - Scripts/sfcr-app/src/assets/vite.svg


+ 43 - 0
04 - Scripts/sfcr-app/src/constants.js

@@ -0,0 +1,43 @@
+export const ALL_COMPANIES = [
+  { id: 1, name: "Abeille Epargne Retraite", logo: "AE" },
+  { id: 2, name: "Abeille Vie", logo: "AV" },
+  { id: 3, name: "ACM Vie SA", logo: "AC" },
+  { id: 4, name: "ACM Vie SAM", logo: "AM" },
+  { id: 5, name: "Allianz Vie", logo: "AL" },
+  { id: 6, name: "ANTARIUS", logo: "AN" },
+  { id: 7, name: "AXA Assurances Vie Mutuelle", logo: "AX" },
+  { id: 8, name: "AXA France Vie", logo: "AF" },
+  { id: 9, name: "BPCE VIE", logo: "BP" },
+  { id: 10, name: "Caisse Générale de Prévoyance (CGP)", logo: "CG" },
+  { id: 11, name: "Cardif Assurance Vie", logo: "CA" },
+  { id: 12, name: "CNP Assurances", logo: "CN" },
+  { id: 13, name: "Generali Vie", logo: "GE" },
+  { id: 14, name: "GMF Vie", logo: "GM" },
+  { id: 15, name: "Groupama Gan Vie", logo: "GG" },
+  { id: 16, name: "HSBC Assurances Vie", logo: "HS" },
+  { id: 17, name: "La France Mutualiste", logo: "LF" },
+  { id: 18, name: "MAAF Vie", logo: "MF" },
+  { id: 19, name: "MACSF Epargne Retraite", logo: "MC" },
+  { id: 20, name: "MAIF Vie", logo: "MA" },
+  { id: 21, name: "MMA Vie SA", logo: "MM" },
+  { id: 22, name: "Prépar Vie", logo: "PR" },
+  { id: 23, name: "SMA Vie BTP", logo: "SM" },
+  { id: 24, name: "SOGECAP", logo: "SO" },
+  { id: 25, name: "SPIRICA", logo: "SP" },
+  { id: 26, name: "SURAVENIR SA", logo: "SU" },
+  { id: 27, name: "SwissLife Assurance et Patrimoine", logo: "SW" },
+];
+
+export const SFCR_SECTIONS = ["S.02", "S.05", "S.12", "S.17", "S.22", "S.25", "S.28"];
+export const YEARS = [2025, 2024, 2023, 2022, 2021, 2020, 2019];
+export const CURRENT_YEAR = 2025;
+
+export const sectionLabels = {
+  "S.02": "Bilan",
+  "S.05": "Primes, sinistres et dépenses",
+  "S.12": "Provisions techniques vie",
+  "S.17": "Provisions techniques non-vie",
+  "S.22": "Impact mesures long terme",
+  "S.25": "Capital de solvabilité requis",
+  "S.28": "MCR",
+};

+ 10 - 0
04 - Scripts/sfcr-app/src/index.css

@@ -0,0 +1,10 @@
+* {
+  margin: 0;
+  padding: 0;
+  box-sizing: border-box;
+}
+
+html, body, #root {
+  width: 100%;
+  height: 100%;
+}

+ 10 - 0
04 - Scripts/sfcr-app/src/main.jsx

@@ -0,0 +1,10 @@
+import { StrictMode } from 'react'
+import { createRoot } from 'react-dom/client'
+import './index.css'
+import App from './App.jsx'
+
+createRoot(document.getElementById('root')).render(
+  <StrictMode>
+    <App />
+  </StrictMode>,
+)

+ 587 - 0
04 - Scripts/sfcr-app/src/styles.js

@@ -0,0 +1,587 @@
+export const C = {
+  bg: "#0d0f14",
+  surface: "#141720",
+  surfaceHover: "#1c2030",
+  border: "#252a38",
+  borderLight: "#2e3448",
+  accent: "#3b82f6",
+  accentGlow: "rgba(59,130,246,0.15)",
+  accentSoft: "rgba(59,130,246,0.08)",
+  gold: "#f59e0b",
+  goldSoft: "rgba(245,158,11,0.12)",
+  text: "#e8ecf4",
+  textMuted: "#6b7594",
+  textDim: "#9aa3bc",
+  green: "#10b981",
+  greenSoft: "rgba(16,185,129,0.1)",
+};
+
+export const styles = {
+
+    extractionWrapper: {
+    padding: '30px',
+    animation: 'fadeIn 0.4s ease-out',
+  },
+  extractionHeader: {
+    display: 'flex',
+    justifyContent: 'space-between',
+    alignItems: 'center',
+    marginBottom: '25px',
+  },
+  extractionList: {
+    display: 'flex',
+    flexDirection: 'column',
+    gap: '15px',
+  },
+  extractionCard: {
+    background: C.surface,
+    border: `1px solid ${C.border}`,
+    borderRadius: '12px',
+    padding: '20px',
+    boxShadow: '0 4px 20px rgba(0,0,0,0.2)',
+  },
+  companyStatusHeader: {
+    display: 'flex',
+    alignItems: 'center',
+    gap: '15px',
+    marginBottom: '20px',
+    borderBottom: `1px solid ${C.border}`,
+    paddingBottom: '15px',
+  },
+  stepsTimeline: {
+    display: 'flex',
+    justifyContent: 'space-between',
+    padding: '0 10px',
+  },
+  stepItemActive: {
+    display: 'flex',
+    flexDirection: 'column',
+    alignItems: 'center',
+    gap: '8px',
+    fontSize: '11px',
+    color: C.accent,
+    fontWeight: 'bold',
+  },
+  stepItemPending: {
+    display: 'flex',
+    flexDirection: 'column',
+    alignItems: 'center',
+    gap: '8px',
+    fontSize: '11px',
+    color: C.textMuted,
+    opacity: 0.6,
+  },
+  stepIcon: {
+    fontSize: '18px',
+  },
+
+
+  root: {
+    display: "flex",
+    height: "100vh",
+    background: C.bg,
+    color: C.text,
+    fontFamily: "'DM Sans', 'Segoe UI', sans-serif",
+    overflow: "hidden",
+  },
+  sidebar: {
+    background: C.surface,
+    borderRight: `1px solid ${C.border}`,
+    display: "flex",
+    flexDirection: "column",
+    transition: "width 0.25s cubic-bezier(.4,0,.2,1)",
+    flexShrink: 0,
+    overflow: "hidden",
+  },
+  sidebarHeader: {
+    display: "flex",
+    alignItems: "center",
+    justifyContent: "space-between",
+    padding: "20px 16px 12px",
+    borderBottom: `1px solid ${C.border}`,
+    minHeight: 64,
+  },
+  brandText: {
+    display: "flex",
+    alignItems: "baseline",
+    gap: 4,
+    whiteSpace: "nowrap",
+  },
+  brandAccent: {
+    fontSize: 18,
+    fontWeight: 800,
+    letterSpacing: "-0.5px",
+    color: C.accent,
+  },
+  brandSub: {
+    fontSize: 13,
+    color: C.textMuted,
+    fontWeight: 400,
+  },
+  collapseBtn: {
+    background: "none",
+    border: `1px solid ${C.border}`,
+    color: C.textDim,
+    width: 28,
+    height: 28,
+    borderRadius: 6,
+    cursor: "pointer",
+    fontSize: 16,
+    display: "flex",
+    alignItems: "center",
+    justifyContent: "center",
+    flexShrink: 0,
+  },
+  nav: {
+    padding: "12px 10px",
+    display: "flex",
+    flexDirection: "column",
+    gap: 4,
+    flex: 1,
+  },
+  navBtn: {
+    display: "flex",
+    alignItems: "center",
+    gap: 10,
+    padding: "10px 12px",
+    borderRadius: 8,
+    border: "none",
+    background: "none",
+    color: C.textMuted,
+    cursor: "pointer",
+    textAlign: "left",
+    fontSize: 13.5,
+    fontWeight: 500,
+    transition: "all 0.15s",
+    whiteSpace: "nowrap",
+  },
+  navBtnActive: {
+    background: C.accentSoft,
+    color: C.accent,
+    borderLeft: `2px solid ${C.accent}`,
+  },
+  navIcon: { fontSize: 16, flexShrink: 0 },
+  navLabel: { overflow: "hidden", textOverflow: "ellipsis" },
+  sidebarFooter: {
+    padding: "12px 16px 20px",
+    borderTop: `1px solid ${C.border}`,
+  },
+  footerBadge: {
+    display: "inline-block",
+    fontSize: 10,
+    fontWeight: 700,
+    letterSpacing: "1px",
+    color: C.gold,
+    background: C.goldSoft,
+    border: `1px solid rgba(245,158,11,0.2)`,
+    borderRadius: 4,
+    padding: "3px 8px",
+    textTransform: "uppercase",
+  },
+  main: {
+    flex: 1,
+    overflow: "auto",
+    background: C.bg,
+  },
+  placeholderPanel: {
+    display: "flex",
+    flexDirection: "column",
+    alignItems: "center",
+    justifyContent: "center",
+    height: "100%",
+    gap: 16,
+    color: C.textMuted,
+  },
+  placeholderIcon: { fontSize: 48, opacity: 0.3 },
+  placeholderTitle: { fontSize: 22, fontWeight: 600, color: C.textDim, margin: 0 },
+  placeholderDesc: { fontSize: 14, margin: 0 },
+  extractionPanel: {
+    padding: "28px 32px",
+    minHeight: "100%",
+    boxSizing: "border-box",
+  },
+  pageHeader: {
+    display: "flex",
+    justifyContent: "space-between",
+    alignItems: "flex-start",
+    marginBottom: 28,
+  },
+  pageTitle: {
+    fontSize: 26,
+    fontWeight: 800,
+    margin: "0 0 4px",
+    letterSpacing: "-0.5px",
+    background: `linear-gradient(135deg, ${C.text}, ${C.textDim})`,
+    WebkitBackgroundClip: "text",
+    WebkitTextFillColor: "transparent",
+  },
+  pageSubtitle: { fontSize: 13, color: C.textMuted, margin: 0 },
+  extractBtn: {
+    background: C.accent,
+    color: "#fff",
+    border: "none",
+    borderRadius: 8,
+    padding: "11px 22px",
+    fontSize: 14,
+    fontWeight: 600,
+    cursor: "pointer",
+    display: "flex",
+    alignItems: "center",
+    gap: 8,
+    transition: "all 0.2s",
+    boxShadow: `0 0 24px ${C.accentGlow}`,
+  },
+  twoCol: {
+    display: "grid",
+    gridTemplateColumns: "1fr 1fr",
+    gap: 20,
+    alignItems: "start",
+  },
+  leftCol: { display: "flex", flexDirection: "column", gap: 20 },
+  rightCol: { display: "flex", flexDirection: "column", gap: 20 },
+  card: {
+    background: C.surface,
+    border: `1px solid ${C.border}`,
+    borderRadius: 12,
+    padding: "20px 22px",
+  },
+  cardHeader: {
+    display: "flex",
+    alignItems: "center",
+    gap: 10,
+    marginBottom: 16,
+  },
+  cardBadge: {
+    fontSize: 11,
+    fontWeight: 800,
+    color: C.accent,
+    background: C.accentSoft,
+    border: `1px solid rgba(59,130,246,0.25)`,
+    borderRadius: 5,
+    padding: "2px 7px",
+    letterSpacing: "0.5px",
+  },
+  cardTitle: {
+    fontSize: 15,
+    fontWeight: 700,
+    margin: 0,
+    flex: 1,
+  },
+  countPill: {
+    fontSize: 11,
+    fontWeight: 600,
+    color: C.green,
+    background: C.greenSoft,
+    borderRadius: 20,
+    padding: "2px 10px",
+  },
+  searchRow: {
+    display: "flex",
+    alignItems: "center",
+    gap: 8,
+    background: C.bg,
+    border: `1px solid ${C.border}`,
+    borderRadius: 8,
+    padding: "8px 12px",
+    marginBottom: 10,
+  },
+  searchIcon: { fontSize: 13, opacity: 0.5, flexShrink: 0 },
+  searchInput: {
+    background: "none",
+    border: "none",
+    outline: "none",
+    color: C.text,
+    fontSize: 13,
+    flex: 1,
+  },
+  selectAllRow: {
+    marginBottom: 8,
+    paddingBottom: 8,
+    borderBottom: `1px solid ${C.border}`,
+  },
+  checkLabel: {
+    display: "flex",
+    alignItems: "center",
+    gap: 8,
+    cursor: "pointer",
+  },
+  hiddenCheck: { display: "none" },
+  customCheck: {
+    width: 16,
+    height: 16,
+    border: `1.5px solid ${C.borderLight}`,
+    borderRadius: 4,
+    display: "flex",
+    alignItems: "center",
+    justifyContent: "center",
+    fontSize: 10,
+    fontWeight: 700,
+    flexShrink: 0,
+    transition: "all 0.15s",
+  },
+  customCheckChecked: {
+    background: C.accent,
+    borderColor: C.accent,
+    color: "#fff",
+  },
+  checkLabelText: { fontSize: 13, color: C.textDim },
+  companyList: { display: "flex", flexDirection: "column", gap: 2 },
+  companyRow: {
+    display: "flex",
+    alignItems: "center",
+    gap: 10,
+    padding: "8px 10px",
+    borderRadius: 7,
+    cursor: "pointer",
+    border: `1px solid transparent`,
+    transition: "all 0.15s",
+  },
+  companyRowChecked: {
+    background: C.accentSoft,
+    border: `1px solid rgba(59,130,246,0.2)`,
+  },
+  companyAvatar: {
+    width: 28,
+    height: 28,
+    borderRadius: 6,
+    background: `linear-gradient(135deg, #1e3a5f, #2d5a8e)`,
+    display: "flex",
+    alignItems: "center",
+    justifyContent: "center",
+    fontSize: 10,
+    fontWeight: 700,
+    color: "#7fb3e8",
+    flexShrink: 0,
+  },
+  companyName: { fontSize: 13, flex: 1, fontWeight: 450 },
+  showMoreBtn: {
+    marginTop: 12,
+    background: "none",
+    border: `1px dashed ${C.borderLight}`,
+    color: C.textMuted,
+    borderRadius: 7,
+    padding: "8px 14px",
+    fontSize: 12,
+    cursor: "pointer",
+    width: "100%",
+    transition: "all 0.15s",
+  },
+  modeToggle: {
+    display: "flex",
+    gap: 6,
+    marginBottom: 14,
+    background: C.bg,
+    borderRadius: 8,
+    padding: 4,
+  },
+  modeBtn: {
+    flex: 1,
+    background: "none",
+    border: "none",
+    color: C.textMuted,
+    borderRadius: 5,
+    padding: "7px 10px",
+    fontSize: 12,
+    cursor: "pointer",
+    fontWeight: 500,
+    transition: "all 0.15s",
+  },
+  modeBtnActive: {
+    background: C.surfaceHover,
+    color: C.text,
+    boxShadow: `0 1px 3px rgba(0,0,0,0.3)`,
+  },
+  sectionGrid: {
+    display: "grid",
+    gridTemplateColumns: "1fr 1fr",
+    gap: 8,
+  },
+  sectionChip: {
+    display: "flex",
+    flexDirection: "column",
+    alignItems: "flex-start",
+    padding: "10px 12px",
+    borderRadius: 8,
+    border: `1px solid ${C.border}`,
+    background: C.bg,
+    cursor: "pointer",
+    transition: "all 0.15s",
+    textAlign: "left",
+    gap: 3,
+  },
+  sectionChipActive: {
+    background: C.accentSoft,
+    border: `1px solid rgba(59,130,246,0.35)`,
+  },
+  sectionCode: {
+    fontSize: 13,
+    fontWeight: 800,
+    color: C.accent,
+    letterSpacing: "0.3px",
+  },
+  sectionLabel: { fontSize: 10.5, color: C.textMuted, lineHeight: 1.3 },
+  yearGrid: {
+    display: "flex",
+    gap: 8,
+    flexWrap: "wrap",
+  },
+  yearChip: {
+    padding: "8px 18px",
+    borderRadius: 7,
+    border: `1px solid ${C.border}`,
+    background: C.bg,
+    color: C.textDim,
+    fontSize: 13,
+    fontWeight: 600,
+    cursor: "pointer",
+    transition: "all 0.15s",
+  },
+  yearChipActive: {
+    background: C.goldSoft,
+    border: `1px solid rgba(245,158,11,0.35)`,
+    color: C.gold,
+  },
+  perCompanyList: { display: "flex", flexDirection: "column", gap: 12 },
+  perCompanyRow: {
+    display: "flex",
+    alignItems: "center",
+    gap: 10,
+    flexWrap: "wrap",
+  },
+  perCompanyBlock: {
+    background: C.bg,
+    borderRadius: 8,
+    padding: "10px 12px",
+    border: `1px solid ${C.border}`,
+  },
+  perCompanyHeader: {
+    display: "flex",
+    alignItems: "center",
+    gap: 8,
+    marginBottom: 10,
+  },
+  perCompanyAvatar: {
+    width: 24,
+    height: 24,
+    borderRadius: 5,
+    background: `linear-gradient(135deg, #1e3a5f, #2d5a8e)`,
+    display: "flex",
+    alignItems: "center",
+    justifyContent: "center",
+    fontSize: 9,
+    fontWeight: 700,
+    color: "#7fb3e8",
+    flexShrink: 0,
+  },
+  perCompanyName: { fontSize: 12, color: C.textDim, fontWeight: 500 },
+  miniYearRow: { display: "flex", gap: 5, flexWrap: "wrap" },
+  miniYearChip: {
+    padding: "4px 10px",
+    borderRadius: 5,
+    border: `1px solid ${C.border}`,
+    background: "none",
+    color: C.textMuted,
+    fontSize: 11,
+    cursor: "pointer",
+    fontWeight: 600,
+    transition: "all 0.15s",
+  },
+  miniYearChipActive: {
+    background: C.goldSoft,
+    borderColor: "rgba(245,158,11,0.35)",
+    color: C.gold,
+  },
+  miniSectionGrid: { display: "flex", gap: 5, flexWrap: "wrap" },
+  miniSectionChip: {
+    padding: "4px 9px",
+    borderRadius: 5,
+    border: `1px solid ${C.border}`,
+    background: "none",
+    color: C.textMuted,
+    fontSize: 11,
+    fontWeight: 700,
+    cursor: "pointer",
+    transition: "all 0.15s",
+    letterSpacing: "0.3px",
+  },
+  miniSectionChipActive: {
+    background: C.accentSoft,
+    borderColor: "rgba(59,130,246,0.35)",
+    color: C.accent,
+  },
+  promptWrapper: {
+    background: C.bg,
+    border: `1px solid ${C.border}`,
+    borderRadius: 8,
+    overflow: "hidden",
+    transition: "border-color 0.15s",
+  },
+  promptInput: {
+    width: "100%",
+    background: "none",
+    border: "none",
+    outline: "none",
+    color: C.text,
+    fontSize: 13,
+    padding: "12px 14px",
+    resize: "none",
+    fontFamily: "inherit",
+    lineHeight: 1.6,
+    boxSizing: "border-box",
+  },
+  promptFooter: {
+    display: "flex",
+    justifyContent: "space-between",
+    alignItems: "center",
+    padding: "6px 12px",
+    borderTop: `1px solid ${C.border}`,
+  },
+  promptHint: { fontSize: 11, color: C.textMuted },
+  clearBtn: {
+    background: "none",
+    border: "none",
+    color: C.textMuted,
+    fontSize: 11,
+    cursor: "pointer",
+    padding: "2px 6px",
+  },
+  summaryCard: {
+    background: `linear-gradient(135deg, rgba(59,130,246,0.06), rgba(16,185,129,0.06))`,
+    border: `1px solid rgba(59,130,246,0.2)`,
+    borderRadius: 12,
+    padding: "16px 20px",
+  },
+  summaryTitle: {
+    fontSize: 12,
+    fontWeight: 700,
+    color: C.textMuted,
+    textTransform: "uppercase",
+    letterSpacing: "1px",
+    margin: "0 0 14px",
+  },
+  summaryGrid: { display: "flex", gap: 0 },
+  summaryItem: {
+    flex: 1,
+    display: "flex",
+    flexDirection: "column",
+    alignItems: "center",
+    gap: 3,
+    borderRight: `1px solid ${C.border}`,
+    padding: "0 10px",
+  },
+  summaryNum: { fontSize: 28, fontWeight: 800, color: C.accent, lineHeight: 1 },
+  summaryItemLabel: { fontSize: 11, color: C.textMuted },
+};
+
+export const cssString = `
+  @import url('https://fonts.googleapis.com/css2?family=DM+Sans:wght@400;500;600;700;800&display=swap');
+  * { box-sizing: border-box; }
+  body { margin: 0; }
+  ::-webkit-scrollbar { width: 5px; height: 5px; }
+  ::-webkit-scrollbar-track { background: transparent; }
+  ::-webkit-scrollbar-thumb { background: #252a38; border-radius: 99px; }
+  .nav-btn:hover { background: #1c2030 !important; color: #9aa3bc !important; }
+  .company-row:hover { background: #1c2030 !important; }
+  .section-chip:hover { background: #1c2030 !important; }
+  .extract-btn:hover { opacity: 0.9; transform: translateY(-1px); }
+`;

+ 7 - 0
04 - Scripts/sfcr-app/vite.config.js

@@ -0,0 +1,7 @@
+import { defineConfig } from 'vite'
+import react from '@vitejs/plugin-react'
+
+// https://vite.dev/config/
+export default defineConfig({
+  plugins: [react()],
+})

+ 35 - 0
04 - Scripts/test_luncher.py

@@ -0,0 +1,35 @@
+import os
+import sys
+import subprocess
+from pathlib import Path
+
+# On récupère le chemin absolu du dossier actuel
+current_dir = Path(__file__).resolve().parent
+main_path = current_dir / "main.py"
+
+print(f"--- DIAGNOSTIC ---")
+print(f"Dossier actuel : {current_dir}")
+print(f"Chemin de main.py : {main_path}")
+print(f"Est-ce que main.py existe ? : {main_path.exists()}")
+print(f"Interpréteur Python utilisé : {sys.executable}")
+print(f"------------------\n")
+
+print("Tentative de lancement de main.py...")
+
+try:
+    # On lance main.py et on attend qu'il finisse
+    result = subprocess.run(
+        [sys.executable, str(main_path)],
+        capture_output=True,
+        text=True,
+        cwd=str(current_dir)
+    )
+
+    print("--- RÉSULTAT DU LANCEMENT ---")
+    print(f"Code de sortie (0 = OK) : {result.returncode}")
+    print(f"Sortie standard (STDOUT) :\n{result.stdout}")
+    print(f"Erreurs (STDERR) :\n{result.stderr}")
+    print("-----------------------------")
+
+except Exception as e:
+    print(f"ERREUR lors de l'exécution : {e}")

+ 42 - 0
04 - Scripts/tools.py

@@ -0,0 +1,42 @@
+import os
+import pandas as pd
+from langchain_core.tools import tool
+
+@tool
+def excel_code_interpreter(code: str, entreprise_name: str):
+    """
+    Exécute du code Python pour générer un tableau Excel.
+    Le fichier sera enregistré dans un dossier au nom de l'entreprise.
+    Arguments:
+    - code: Le code Python à exécuter (doit utiliser pandas).
+    - entreprise_name: Le nom de l'entreprise pour créer le dossier.
+    """
+    import warnings
+    warnings.filterwarnings("ignore")
+    
+    try:
+        # 1. Création du dossier de l'entreprise s'il n'existe pas
+        # On nettoie le nom pour éviter les problèmes de caractères spéciaux
+        folder_path = entreprise_name.replace(" ", "_").strip()
+        if not os.path.exists(folder_path):
+            os.makedirs(folder_path)
+
+        # 2. Préparation du contexte d'exécution
+        # On définit le chemin de sortie par défaut pour l'agent
+        output_file = os.path.join(folder_path, f"Rapport_{folder_path}.xlsx")
+        
+        context = {
+            "pd": pd,
+            "os": os,
+            "output_file": output_file,
+            "result": None
+        }
+
+        # 3. Exécution du code
+        # L'agent doit utiliser 'pd.ExcelWriter(output_file, engine="xlsxwriter")' dans son code
+        exec(code, context)
+        
+        return f"✅ Succès : Code exécuté. Fichier enregistré dans : {output_file}"
+
+    except Exception as e:
+        return f"❌ ERREUR PYTHON : {str(e)}"

+ 44 - 0
04 - Scripts/workflow_agents.py

@@ -0,0 +1,44 @@
+from langgraph.graph import StateGraph, START, END
+
+# ⚠️ Import corrigé (selon version)
+try:
+    from langgraph.prebuilt import ToolNode
+except ImportError:
+    # fallback si ToolNode n'existe pas
+    from langchain_core.runnables import RunnableLambda
+
+    def ToolNode(tools):
+        def run_tools(state):
+            return state  # à adapter si besoin
+        return RunnableLambda(run_tools)
+
+# Assure-toi que ces imports sont corrects
+from Agents import AgentState, agent_extracteur, agent_builder, agent_ocr, tools
+
+workflow = StateGraph(AgentState)
+
+# 1. Définition des Nœuds
+workflow.add_node("agent_ocr", agent_ocr)
+workflow.add_node("agent_extracteur", agent_extracteur)
+workflow.add_node("agent_builder", agent_builder)
+workflow.add_node("tools", ToolNode(tools))
+
+# 2. Définition des Arêtes
+workflow.add_edge(START, "agent_ocr")
+workflow.add_edge("agent_ocr", "agent_extracteur")
+workflow.add_edge("agent_extracteur", "agent_builder")
+
+
+workflow.add_edge("tools",END)
+
+
+# 5. Compilation
+app = workflow.compile() 
+
+# 6. Graph
+try:
+    with open("graph_workflow.png", "wb") as f:
+        f.write(app.get_graph().draw_mermaid_png())
+    print(" Graphique du workflow généré sous : graph_workflow.png")
+except Exception as e:
+    print(f" Erreur génération image : {e}")

Some files were not shown because too many files changed in this diff