#!/usr/bin/env python3 """ GÉO-RÉVISION — Tectonique des Plaques Application Gradio compatible HuggingFace Spaces 76 notions · CH.3 · CH.4 · CH.5 · Synthèse """ import gradio as gr import random import re # ══════════════════════════════════════════════════════════ # DONNÉES — 76 cartes # ══════════════════════════════════════════════════════════ CARDS = [ {"id": 1, "ch": "3", "section": "CH.3 — Limites de plaques", "q": "Quels sont les 3 grands types de limites de plaques ?", "hint": "💡 Pense : naissance / destruction / collision", "a": "**1. DORSALES** (divergence / accrétion océanique)\n**2. FOSSES OCÉANIQUES** (convergence / subduction)\n**3. CHAÎNES DE MONTAGNES** (convergence / collision)"}, {"id": 2, "ch": "3", "section": "CH.3 — Dorsales", "q": "Quelles roches sont produites à la dorsale ? Quelle est leur texture ?", "hint": "💡 Deux roches, deux textures", "a": "**BASALTE** : texture microlithique (refroidissement *rapide* en surface)\n**GABBRO** : texture grenue (refroidissement *lent* en profondeur)"}, {"id": 3, "ch": "3", "section": "CH.3 — Dorsales", "q": "Quel type de failles caractérise les dorsales ? Pourquoi ?", "hint": "💡 Extension ou compression ?", "a": "**FAILLES NORMALES** car la divergence crée une **EXTENSION** (étirement) de la croûte."}, {"id": 4, "ch": "3", "section": "CH.3 — Dorsales", "q": "Caractéristiques sismiques et thermiques d'une dorsale ?", "hint": "💡 Profondeur + intensité + flux", "a": "Sismicité **SUPERFICIELLE** et **FAIBLE MAGNITUDE**.\nFlux géothermique **ÉLEVÉ** (magmatisme ascendant)."}, {"id": 5, "ch": "3", "section": "CH.3 — Fosses", "q": "Quelles roches plutoniques et volcaniques sont produites en subduction ?", "hint": "💡 4 roches, 2 textures différentes", "a": "Plutoniques (grenues) : **DIORITE**, **GRANITE**\nVolcaniques (microlithiques) : **ANDÉSITE**, **RHYOLITE**"}, {"id": 6, "ch": "3", "section": "CH.3 — Fosses", "q": "Comment se répartissent les foyers sismiques en zone de subduction ? Jusqu'où ?", "hint": "💡 Un plan célèbre...", "a": "Plan de **WADATI-BENIOFF** : plan incliné suivant la plaque plongeante.\nAtteint **700 KM** de profondeur."}, {"id": 7, "ch": "3", "section": "CH.3 — Fosses", "q": "Décris le flux géothermique en zone de subduction (deux zones).", "hint": "💡 Froid ici, chaud là-bas", "a": "**BAS** au niveau de la plaque froide qui plonge.\n**FORT** au niveau de l'arc volcanique sus-jacent."}, {"id": 8, "ch": "3", "section": "CH.3 — Collision", "q": "Caractéristiques géologiques et sismiques d'une zone de collision ?", "hint": "💡 Type de failles + profondeur séismes", "a": "**FAILLES INVERSES**, flux géothermique **FAIBLE**,\nsismicité principalement **SUPERFICIELLE**."}, {"id": 9, "ch": "3", "section": "CH.3 — Points chauds", "q": "Comment un alignement de volcans inactifs prouve-t-il le mouvement des plaques ?", "hint": "💡 Âge + distance", "a": "Le point chaud est **FIXE** dans le manteau profond.\nL'âge des volcans **AUGMENTE avec la distance** au volcan actif."}, {"id": 10, "ch": "3", "section": "CH.3 — Anomalies magnétiques", "q": "Qu'est-ce qu'une anomalie magnétique positive vs négative ?", "hint": "💡 Intensité mesurée vs moyenne", "a": "**POSITIVE** : intensité > moyenne → polarité **NORMALE**\n**NÉGATIVE** : intensité < moyenne → polarité **INVERSE** (pôles inversés)"}, {"id": 11, "ch": "3", "section": "CH.3 — Anomalies magnétiques", "q": "Décris le patron des anomalies magnétiques autour d'une dorsale.", "hint": "💡 Symétrie + image", "a": "Bandes **PARALLÈLES ET SYMÉTRIQUES** par rapport à l'axe de la dorsale.\nComme un **DOUBLE TAPIS ROULANT**.\nBasaltes d'autant plus vieux qu'ils sont éloignés."}, {"id": 12, "ch": "3", "section": "CH.3 — Anomalies magnétiques", "q": "Que révèle l'âge et l'épaisseur des sédiments sur le plancher océanique ?", "hint": "💡 Loi simple distance/âge", "a": "Épaisseur et âge des sédiments **AUGMENTENT** avec la distance à la dorsale.\n→ Preuve de l'**expansion océanique**."}, {"id": 13, "ch": "3", "section": "CH.3 — Géodésie spatiale", "q": "Que permet la géodésie spatiale (GPS) en tectonique ?", "hint": "💡 Précision + deux types de vitesses", "a": "Localisation au **MILLIMÈTRE PRÈS**.\nCalcul des vitesses **ABSOLUES** (repères fixes) et **RELATIVES** (entre deux plaques)."}, {"id": 14, "ch": "3", "section": "CH.3 — Résumé frontières", "q": "Quelle frontière présente une vallée axiale (rift) ? Quel mouvement ?", "hint": "💡 Rift = fossé d'effondrement", "a": "La **DORSALE** : mouvement de **DIVERGENCE**.\nLa vallée axiale (rift) témoigne de l'extension de la croûte."}, {"id": 15, "ch": "3", "section": "CH.3 — Structure générale", "q": "Lithosphère terrestre : structure de base et activité aux frontières.", "hint": "💡 Découpe + concentration", "a": "Découpée en **PLAQUES LITHOSPHÉRIQUES RIGIDES**.\nFrontières concentrent l'**ACTIVITÉ SISMIQUE ET VOLCANIQUE**."}, {"id": 16, "ch": "3", "section": "CH.3 — Anomalies magnétiques", "q": "Comment les basaltes enregistrent-ils le champ magnétique ?", "hint": "💡 Moment clé = refroidissement", "a": "En se **REFROIDISSANT**, les minéraux magnétiques (magnétite) se figent dans la direction du champ magnétique terrestre du moment."}, {"id": 17, "ch": "3", "section": "CH.3 — Dorsales", "q": "Que produit l'accrétion océanique à la dorsale ?", "hint": "💡 Naît quoi ?", "a": "La **CROÛTE OCÉANIQUE** :\n- Basalte en surface\n- Gabbro en profondeur\n- Péridotite mantellique sous-jacente"}, {"id": 18, "ch": "3", "section": "CH.3 — Fosses", "q": "Qu'est-ce qu'une fosse océanique ? Quel processus y est associé ?", "hint": "💡 Plongement...", "a": "Dépression profonde marquant une **ZONE DE SUBDUCTION** :\nplongement d'une plaque océanique dense dans le manteau."}, {"id": 19, "ch": "3", "section": "CH.3 — Collision", "q": "Qu'est-ce qu'une chaîne de montagnes en termes tectoniques ?", "hint": "💡 Fermeture d'un océan", "a": "Zone de **COLLISION CONTINENTALE** : résultat de la convergence de deux plaques continentales après fermeture d'un océan."}, {"id": 20, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Quelle est la vitesse d'écartement d'une dorsale rapide ?", "hint": "💡 10–16 cm/an — Quel océan ?", "a": "**10 à 16 CM/AN**.\nExemple : dorsale du **Pacifique Est** (Pacifique)."}, {"id": 21, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Morphologie d'une dorsale rapide : rift, bombement ?", "hint": "💡 Opposé de la dorsale lente", "a": "**BOMBEMENT LARGE** mais **VALLÉE AXIALE QUASI ABSENTE** (magmatisme abondant comble le rift)."}, {"id": 22, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Décris le processus de fusion partielle sous une dorsale (3 étapes).", "hint": "💡 Convection → décompression → solidus", "a": "**1.** Remontée des péridotites par **CONVECTION**\n**2.** **DÉCOMPRESSION ADIABATIQUE** : le géotherme recoupe le solidus\n**3.** **FUSION PARTIELLE** (~15%) → gouttelettes de magma"}, {"id": 23, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Où se loge la chambre magmatique ? Quelles roches en sortent ?", "hint": "💡 Profondeur + 2 roches", "a": "Entre **2 et 7 KM** de profondeur.\nRefroidissement lent → **GABBROS** (grenu)\nRefroidissement rapide → **BASALTES** (microlithique)"}, {"id": 24, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Quelle épaisseur de croûte produit une dorsale rapide ?", "hint": "💡 Chiffre précis en km", "a": "Croûte océanique régulière de **5 À 7 KM** d'épaisseur."}, {"id": 25, "ch": "4", "section": "CH.4 — Dorsales lentes", "q": "Quelle est la vitesse d'écartement d'une dorsale lente ?", "hint": "💡 1–5 cm/an — Quel océan ?", "a": "**1 à 5 CM/AN**.\nExemple : dorsale **médio-atlantique** (Atlantique)."}, {"id": 26, "ch": "4", "section": "CH.4 — Dorsales lentes", "q": "Morphologie d'une dorsale lente : rift, magmatisme, expansion.", "hint": "💡 Opposé de rapide", "a": "**RIFT TRÈS MARQUÉ** + grandes failles de détachement.\nMagmatisme **RÉDUIT ET DISCONTINU**.\nExpansion parfois assurée par **ÉTIREMENT TECTONIQUE**."}, {"id": 27, "ch": "4", "section": "CH.4 — Dorsales lentes", "q": "Qu'est-ce que la serpentinisation et quand se produit-elle ?", "hint": "💡 Roche + eau + dorsale lente", "a": "Hydratation des **PÉRIDOTITES** (manteau) remontées en surface.\nL'eau transforme l'olivine en **SERPENTINE**."}, {"id": 28, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Qu'est-ce que le métamorphisme hydrothermal ?", "hint": "💡 Eau + chaleur + minéraux", "a": "Transformation **À L'ÉTAT SOLIDE** des minéraux de la croûte océanique par infiltration d'eau de mer chauffée. Produit des minéraux **HYDRATÉS**."}, {"id": 29, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Séquence de faciès métamorphiques lors du vieillissement océanique ?", "hint": "💡 Deux températures, deux faciès", "a": "**> 700°C** : faciès **AMPHIBOLITE** (hornblende)\n**500–700°C** : faciès **SCHISTE VERT** (chlorite + actinote)"}, {"id": 30, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Comment évolue la densité de la lithosphère océanique avec le temps ?", "hint": "💡 Refroidissement → épaississement → densité", "a": "La lithosphère **SE REFROIDIT**, **S'ÉPAISSIT** (isotherme 1300°C s'approfondit),\nsa **DENSITÉ AUGMENTE** jusqu'à dépasser celle de l'asthénosphère."}, {"id": 31, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Où et comment naissent les fumeurs noirs ?", "hint": "💡 Cycle eau de mer", "a": "Eau de mer s'infiltre, se **RÉCHAUFFE** au contact du magma,\nse charge en minéraux et ressort : **SOURCES HYDROTHERMALES CHAUDES**."}, {"id": 32, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Qu'est-ce que la décompression adiabatique ? Pourquoi déclenche-t-elle la fusion ?", "hint": "💡 Pas de perte de chaleur", "a": "Remontée de matériel **SANS ÉCHANGE DE CHALEUR**.\nPression baisse → solidus baisse → géotherme **RECOUPE LE SOLIDUS** → fusion."}, {"id": 33, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Quelle est la signification de l'isotherme 1300°C pour la lithosphère ?", "hint": "💡 Limite LAB", "a": "**LIMITE LITHOSPHÈRE-ASTHÉNOSPHÈRE (LAB)**.\nElle **S'APPROFONDIT** avec le refroidissement de la plaque éloignée de la dorsale."}, {"id": 34, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Basalte en coussin vs basalte en filon : quelle différence ?", "hint": "💡 Eau vs fissure", "a": "**COUSSIN** : refroidissement très rapide au contact de l'eau (lave sous-marine).\n**FILON (dyke)** : injection de magma dans des fissures verticales."}, {"id": 35, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Quelle minéralogie correspond au faciès schiste vert ?", "hint": "💡 Température < 500°C", "a": "**CHLORITE** + **ACTINOTE** (amphibole calcique) : minéraux hydratés de basse température."}, {"id": 36, "ch": "4", "section": "CH.4 — Dorsales lentes", "q": "Pourquoi une dorsale lente a-t-elle un rift très marqué ?", "hint": "💡 Magma insuffisant pour...", "a": "Magmatisme trop faible pour **COMBLER L'ESPACE** créé par l'écartement.\nL'extension tectonique forme un **FOSSÉ D'EFFONDREMENT PROFOND**."}, {"id": 37, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Transformation minérale lors de l'hydrothermalisme > 700°C ?", "hint": "💡 Faciès amphibolite", "a": "**Pyroxène + Plagioclase** (gabbro) → **HORNBLENDE** (amphibole)\nSous l'action de l'hydrothermalisme à > 700°C."}, {"id": 38, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "La péridotite fond-elle entièrement sous une dorsale ? Quel pourcentage ?", "hint": "💡 Partielle !", "a": "Non : seulement **FUSION PARTIELLE (~15%)**.\nLe reste reste **SOLIDE**. C'est le liquide qui monte former la croûte."}, {"id": 39, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Rôle de la convection mantellique dans le fonctionnement d'une dorsale ?", "hint": "💡 Mouvement ascendant", "a": "La **CONVECTION** assure la **REMONTÉE DES PÉRIDOTITES** sous la dorsale,\ndéclenchant la décompression adiabatique et la fusion partielle."}, {"id": 40, "ch": "4", "section": "CH.4 — Dorsales lentes", "q": "Qu'est-ce qu'une faille de détachement aux dorsales lentes ?", "hint": "💡 Grande faille, faible pendage", "a": "Grande **FAILLE À FAIBLE PENDAGE** (presque horizontale)\npermettant l'exhumation de péridotites directement sur le plancher océanique."}, {"id": 41, "ch": "5", "section": "CH.5 — Marqueurs de subduction", "q": "Quels sont les 3 marqueurs principaux d'une zone de subduction ?", "hint": "💡 1 relief / 1 alignement / 1 sismique", "a": "**1.** FOSSE OCÉANIQUE profonde\n**2.** ALIGNEMENT VOLCANIQUE (arc insulaire ou cordillère)\n**3.** PLAN DE WADATI-BENIOFF (sismicité profonde)"}, {"id": 42, "ch": "5", "section": "CH.5 — Marqueurs de subduction", "q": "Pourquoi la sismicité suit-elle un plan incliné en subduction ?", "hint": "💡 Rigidité de la plaque", "a": "La plaque reste **RIGIDE** en plongeant (elle est froide).\nLes contraintes génèrent des séismes le long du plan de contact : **WADATI-BENIOFF**."}, {"id": 43, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Pourquoi le volcanisme de subduction est-il explosif ?", "hint": "💡 Composition du magma", "a": "Magma riche en **SILICE** (très visqueux) et en **GAZ DISSOUS**.\nViscosité élevée → dégazage brutal → **EXPLOSION VIOLENTE**."}, {"id": 44, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Quelle est la chaîne de réactions menant au volcanisme de subduction ?", "hint": "💡 Eau → péridotite → magma", "a": "**1.** Métamorphisme → libération d'**EAU**\n**2.** Eau monte dans le **COIN MANTELLIQUE**\n**3.** Abaisse le point de fusion des péridotites\n**4.** **FUSION PARTIELLE** → magma → volcanisme"}, {"id": 45, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Quelles sont les 4 roches de subduction et leur texture ?", "hint": "💡 2 volcaniques + 2 plutoniques", "a": "Volcaniques (microlithique) : **ANDÉSITE**, **RHYOLITE**\nPlutoniques (grenue) : **DIORITE**, **GRANITE**"}, {"id": 46, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Pourquoi existe-t-il une telle diversité de roches en subduction ?", "hint": "💡 2 processus", "a": "**CRISTALLISATION FRACTIONNÉE** (différenciation du magma)\n+ **CONTAMINATION** par la croûte continentale traversée."}, {"id": 47, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Quel est le signe minéralogique caractéristique des roches de subduction ?", "hint": "💡 OH dans les minéraux", "a": "Présence de **MINÉRAUX HYDRATÉS** portant des groupements **OH**\n(ex. : amphibole, biotite, muscovite)."}, {"id": 48, "ch": "5", "section": "CH.5 — Métamorphisme", "q": "Séquence de faciès métamorphiques de la plaque subduite ?", "hint": "💡 3 faciès successifs", "a": "**SCHISTE VERT** → **SCHISTE BLEU** (glaucophane, HP-BT) → **ÉCLOGITE** (grenat + jadéite, > 50 km)"}, {"id": 49, "ch": "5", "section": "CH.5 — Métamorphisme", "q": "Quelle minéralogie caractérise le faciès schiste bleu ?", "hint": "💡 Minéral bleu, HP-BT", "a": "**GLAUCOPHANE** (amphibole sodique bleue) :\nindicateur de conditions **HAUTE PRESSION – BASSE TEMPÉRATURE**."}, {"id": 50, "ch": "5", "section": "CH.5 — Métamorphisme", "q": "Minéralogie du faciès éclogite et profondeur de formation ?", "hint": "💡 > 50 km", "a": "**GRENAT** + **JADÉITE** (pyroxène sodique)\nFormés à **> 50 KM** de profondeur, sous très haute pression."}, {"id": 51, "ch": "5", "section": "CH.5 — Métamorphisme", "q": "En quoi le contexte P-T de la subduction est-il unique ?", "hint": "💡 Comparer avec métamorphisme classique", "a": "**HAUTE PRESSION / BASSE TEMPÉRATURE** :\nLa plaque s'enfonce vite et reste froide.\nContraire du métamorphisme de contact (BT/BP)."}, {"id": 52, "ch": "5", "section": "CH.5 — Moteur", "q": "Quel est le moteur principal du mouvement des plaques ?", "hint": "💡 Force de tirage", "a": "**TRACTION DE LA PLAQUE SUBDUITE** (slab pull) :\nPlaque froide et dense → gravité → tire la plaque,\nentraîne les mouvements descendants de la convection mantellique."}, {"id": 53, "ch": "5", "section": "CH.5 — Moteur", "q": "Quel lien y a-t-il entre subduction et convection mantellique ?", "hint": "💡 Cause-conséquence", "a": "La subduction **ENTRETIENT** les courants descendants froids de la convection.\nLes plaques ne sont pas portées passivement : elles **TIRENT ACTIVEMENT**."}, {"id": 54, "ch": "5", "section": "CH.5 — Collision", "q": "Comment se produit une collision continentale ?", "hint": "💡 Après quoi ?", "a": "Après la **FERMETURE COMPLÈTE D'UN OCÉAN** par subduction.\nLes deux marges continentales entrent en contact : croûte trop légère pour plonger."}, {"id": 55, "ch": "5", "section": "CH.5 — Collision", "q": "Quelles structures géologiques témoignent d'une collision en surface ?", "hint": "💡 3 structures", "a": "**PLIS** + **FAILLES INVERSES** + **NAPPES DE CHARRIAGE**\n(grandes masses de roches déplacées horizontalement)"}, {"id": 56, "ch": "5", "section": "CH.5 — Collision", "q": "Qu'est-ce qu'une racine crustale ? À quelle profondeur descend le Moho ?", "hint": "💡 Iceberg crustal", "a": "Épaississement de la croûte vers le bas.\nLe **MOHO** peut descendre jusqu'à **70 KM** (normal : ~35 km)."}, {"id": 57, "ch": "5", "section": "CH.5 — Collision", "q": "Quels types de roches métamorphiques se forment en collision ?", "hint": "💡 Roches foliées", "a": "Roches **FOLIÉES** :\n- **MICASCHISTES** (métamorphisme faible à modéré)\n- **GNEISS** (fort métamorphisme)"}, {"id": 58, "ch": "5", "section": "CH.5 — Marqueurs", "q": "Différence entre arc insulaire et cordillère en subduction ?", "hint": "💡 Océan-Océan vs Océan-Continent", "a": "**ARC INSULAIRE** : subduction Océan/Océan → volcanisme en **MER**.\n**CORDILLÈRE** : subduction Océan/Continent → volcanisme en **BORDURE DE CONTINENT**."}, {"id": 59, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Qu'est-ce que le coin mantellique et quel est son rôle ?", "hint": "💡 Wedge", "a": "Portion du **MANTEAU DE LA PLAQUE CHEVAUCHANTE** entre les deux plaques.\nReçoit l'**EAU libérée** → péridotite fond partiellement → magmatisme."}, {"id": 60, "ch": "5", "section": "CH.5 — Risques", "q": "Risque volcanique caractéristique des zones de subduction ?", "hint": "💡 Nuage mortel", "a": "**NUÉES ARDENTES** (coulées pyroclastiques) :\nmélanges brûlants de gaz, cendres, débris à très grande vitesse.\nLié à la viscosité élevée du magma."}, {"id": 61, "ch": "synth", "section": "SYNTHÈSE — Cycle lithosphérique", "q": "Résume en 5 étapes le cycle de la lithosphère océanique.", "hint": "💡 Naissance → mort", "a": "**1.** Naissance à la **DORSALE** (HT, BP)\n**2.** Hydratation + refroidissement → schistes verts\n**3.** Densification → **SUBDUCTION**\n**4.** Métamorphisme HP-BT (schiste bleu → éclogite) + libération d'eau\n**5.** Déclenchement du **MAGMATISME** → croûte continentale"}, {"id": 62, "ch": "synth", "section": "SYNTHÈSE — Moteur", "q": "Quel processus assure la continuité de la convection mantellique globale ?", "hint": "💡 Traction + plongement", "a": "**TRACTION DE LA PLAQUE PLONGEANTE** (slab pull) :\nforce gravitaire exercée par la plaque froide et dense,\nmoteur principal du cycle convectif global."}, {"id": 63, "ch": "synth", "section": "SYNTHÈSE — Eau", "q": "Quel est le rôle de l'eau dans le cycle de la lithosphère ?", "hint": "💡 Partout : dorsale → subduction", "a": "**DORSALE** : hydratation hydrothermale (schistes verts)\n**SUBDUCTION** : expulsion par métamorphisme → abaissement du solidus → magmatisme"}, {"id": 64, "ch": "synth", "section": "SYNTHÈSE — Cycle", "q": "Où naît et où meurt la lithosphère océanique ?", "hint": "💡 Deux lieux opposés", "a": "**NAISSANCE** : à la **DORSALE** (accrétion)\n**MORT** : à la **FOSSE DE SUBDUCTION** (réintégration dans le manteau)"}, {"id": 65, "ch": "synth", "section": "SYNTHÈSE — Croûte continentale", "q": "Comment la croûte continentale est-elle créée via la subduction ?", "hint": "💡 Eau → fusion → granite", "a": "Eau libérée → fusion coin mantellique → magma riche en silice → volcanisme + intrusions\n→ **GRANITE ET DIORITE** = croûte continentale"}, {"id": 66, "ch": "3", "section": "CH.3 — Anomalies magnétiques", "q": "Qu'est-ce qu'une polarité magnétique normale vs inverse ?", "hint": "💡 Pôles Nord et Sud", "a": "**NORMALE** : orientation actuelle (pôle magnétique Nord ≈ Nord géographique)\n**INVERSE** : pôles **NORD ET SUD MAGNÉTIQUES INVERSÉS**"}, {"id": 67, "ch": "4", "section": "CH.4 — Vieillissement", "q": "Pourquoi la lithosphère océanique devient-elle plus dense avec le temps ?", "hint": "💡 Refroidissement + manteau", "a": "En se refroidissant, elle intègre du manteau refroidi par sa base.\nDensité dépasse celle de l'**ASTHÉNOSPHÈRE** (~3,3 g/cm³)."}, {"id": 68, "ch": "5", "section": "CH.5 — Métamorphisme", "q": "Pourquoi le métamorphisme de subduction est-il HP-BT ?", "hint": "💡 Vitesse d'enfouissement", "a": "La plaque **S'ENFONCE RAPIDEMENT** : pression augmente vite\nmais plaque froide ne se réchauffe pas assez → **HAUTE PRESSION, BASSE TEMPÉRATURE**."}, {"id": 69, "ch": "5", "section": "CH.5 — Collision", "q": "Pourquoi la croûte continentale ne peut-elle pas subduire ?", "hint": "💡 Densité vs manteau", "a": "Croûte continentale : ~**2,7 g/cm³**\nManteau : ~**3,3 g/cm³**\nElle est **MOINS DENSE** → **FLOTTE** → ne peut pas plonger → **COLLISION**."}, {"id": 70, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Qu'est-ce que le géotherme et le solidus dans le contexte des dorsales ?", "hint": "💡 Deux courbes sur un graphique P-T", "a": "**GÉOTHERME** : température en fonction de la profondeur.\n**SOLIDUS** : limite en dessous de laquelle la roche est solide.\nSi géotherme **RECOUPE** le solidus → **LA ROCHE FOND**."}, {"id": 71, "ch": "3", "section": "CH.3 — Preuves expansion", "q": "Quels indices prouvent l'expansion océanique (3 types) ?", "hint": "💡 Magnétique + sédimentaire + géodésique", "a": "**1.** ANOMALIES MAGNÉTIQUES SYMÉTRIQUES\n**2.** ÂGE/ÉPAISSEUR croissants des sédiments\n**3.** MESURES GPS (géodésie spatiale)"}, {"id": 72, "ch": "4", "section": "CH.4 — Dorsales rapides", "q": "Composition minéralogique initiale d'un gabbro (avant métamorphisme) ?", "hint": "💡 2 minéraux principaux", "a": "**PYROXÈNE** (clinopyroxène) + **PLAGIOCLASE** (feldspath calcique).\nRemplacés par hornblende puis chlorite + actinote lors du métamorphisme hydrothermal."}, {"id": 73, "ch": "5", "section": "CH.5 — Moteur", "q": "Quelle force initie le mouvement de la plaque vers la fosse ?", "hint": "💡 Densité + gravité", "a": "Plaque océanique vieillie : **PLUS DENSE QUE L'ASTHÉNOSPHÈRE**.\nGravité → **PLONGEMENT SPONTANÉ** (slab pull) → tire le reste de la plaque."}, {"id": 74, "ch": "5", "section": "CH.5 — Magmatisme", "q": "Quelle roche est l'équivalent plutonique de l'andésite ?", "hint": "💡 Roche à grain visible", "a": "La **DIORITE** : même composition que l'andésite\nmais refroidissement lent en profondeur → texture **GRENUE**."}, {"id": 75, "ch": "4", "section": "CH.4 — Vieillissement", "q": "À quelle température se place la limite faciès amphibolite / faciès schiste vert ?", "hint": "💡 Deux seuils de température", "a": "**AMPHIBOLITE** : > 700°C\n**SCHISTE VERT** : entre 300 et 500°C\nTransition : **500–700°C**"}, {"id": 76, "ch": "synth", "section": "SYNTHÈSE — Les 3 faciès du voyage", "q": "Cite les 3 faciès jalonnant le parcours d'une plaque océanique (dorsale → arc).", "hint": "💡 Dorsale → plaque âgée → subduction", "a": "**1. SCHISTE VERT** (hydrothermalisme dorsale + vieillissement)\n**2. SCHISTE BLEU** (début subduction, HP-BT)\n**3. ÉCLOGITE** (subduction profonde > 50 km)"}, ] WRONG_POOL = [ "Cristallisation fractionnée", "Plan de Wadati-Benioff", "Décompression adiabatique", "Fusion partielle", "Serpentinisation", "Dorsale médio-atlantique", "Glaucophane", "Faille inverse", "Faille normale", "Basalte en coussin", "Jadéite", "Grenat", "Hornblende", "Chlorite", "Actinote", "Gabbro", "Andésite", "Diorite", "Schiste bleu", "Schiste vert", "Éclogite", "Slab pull", "Coin mantellique", "Chambre magmatique", "Isotherme 1300°C", "Solidus", "Géotherme", "Arc insulaire", "Cordillère", "Nuée ardente", "5 à 7 km", "10 à 16 cm/an", "1 à 5 cm/an", "700 km de profondeur", "50 km de profondeur", "Fumeurs noirs", "Micaschiste", "Gneiss", "Péridotite", "Pyroxène", "Plagioclase", "Serpentine", "Nappe de charriage", "Racine crustale", "Rhyolite", "Granite", "Basalte en filon", "Moho à 70 km", ] FILTER_CHOICES = [ "Tout (76)", "CH.3 — Mobilité (19)", "CH.4 — Divergence (21)", "CH.5 — Convergence (20)", "Synthèse (5+)", "⚠ À revoir", ] # ══════════════════════════════════════════════════════════ # LOGIQUE MÉTIER # ══════════════════════════════════════════════════════════ def get_filtered(ch_filter, status_dict): mapping = { "Tout (76)": list(CARDS), "CH.3 — Mobilité (19)": [c for c in CARDS if c["ch"] == "3"], "CH.4 — Divergence (21)":[c for c in CARDS if c["ch"] == "4"], "CH.5 — Convergence (20)":[c for c in CARDS if c["ch"] == "5"], "Synthèse (5+)": [c for c in CARDS if c["ch"] == "synth"], "⚠ À revoir": [c for c in CARDS if status_dict.get(str(c["id"])) in (None, "bad")], } result = mapping.get(ch_filter, list(CARDS)) return result if result else list(CARDS) def ch_badge(ch): return {"3":"🔴 CH.3","4":"🔵 CH.4","5":"🟢 CH.5","synth":"🟡 SYNTH"}.get(ch, ch) def status_emoji(s): return {"good":"✅ Maîtrisé","ok":"🟡 Hésitant","bad":"❌ À revoir"}.get(s, "⬜ Non vu") def compute_stats(status_dict): good = sum(1 for v in status_dict.values() if v == "good") ok = sum(1 for v in status_dict.values() if v == "ok") bad = sum(1 for v in status_dict.values() if v == "bad") unseen = 76 - good - ok - bad pct = int(good / 76 * 100) filled = int(pct / 5) bar = "█" * filled + "░" * (20 - filled) return (f"✅ **{good}** maîtrisées  |  🟡 **{ok}** hésitantes  |  " f"❌ **{bad}** à revoir  |  ⬜ **{unseen}** non vues\n\n" f"`[{bar}]` **{pct}%** complété") # ── Flashcard ── def init_flashcard(ch_filter, state): status_dict = state.get("status", {}) deck = get_filtered(ch_filter, status_dict) random.shuffle(deck) state["deck"] = [c["id"] for c in deck] state["fc_idx"] = 0 state["flipped"] = False return _render_fc(state) def _render_fc(state): deck_ids = state.get("deck", []) idx = state.get("fc_idx", 0) flipped = state.get("flipped", False) status_dict = state.get("status", {}) if not deck_ids: return ("—", "### Aucune carte. Change de filtre.", "", "", "👁 Voir la réponse", gr.update(visible=False), compute_stats(status_dict), state) idx = idx % len(deck_ids) card = next(c for c in CARDS if c["id"] == deck_ids[idx]) s = status_dict.get(str(card["id"])) prog = f"**{idx+1} / {len(deck_ids)}**  ·  {ch_badge(card['ch'])}  ·  {card['section']}  ·  {status_emoji(s)}" q_md = f"## ❓ {card['q']}\n\n{card['hint']}" if not flipped else f"## ❓ {card['q']}" a_md = f"---\n### ✅ Réponse\n\n{card['a']}" if flipped else "" flip_label = "🔄 Masquer la réponse" if flipped else "👁 Voir la réponse" return (prog, q_md, a_md, "", flip_label, gr.update(visible=flipped), compute_stats(status_dict), state) def flip_fc(state): state["flipped"] = not state.get("flipped", False) return _render_fc(state) def next_fc(state): deck_ids = state.get("deck", [c["id"] for c in CARDS]) state["fc_idx"] = (state.get("fc_idx", 0) + 1) % len(deck_ids) state["flipped"] = False return _render_fc(state) def prev_fc(state): deck_ids = state.get("deck", [c["id"] for c in CARDS]) state["fc_idx"] = (state.get("fc_idx", 0) - 1) % len(deck_ids) state["flipped"] = False return _render_fc(state) def rate_fc(rating, state): deck_ids = state.get("deck", [c["id"] for c in CARDS]) idx = state.get("fc_idx", 0) % len(deck_ids) state.setdefault("status", {})[str(deck_ids[idx])] = rating state["fc_idx"] = (idx + 1) % len(deck_ids) state["flipped"] = False return _render_fc(state) def shuffle_fc(state): deck = state.get("deck", [c["id"] for c in CARDS]) random.shuffle(deck) state["deck"] = deck state["fc_idx"] = 0 state["flipped"] = False return _render_fc(state) FC_OUTS = ["prog_md","q_md","a_md","_","flip_btn","rating_row","stats_md","state"] # ── Quiz ── def _build_mcq(card): raw = re.sub(r"\*\*", "", card["a"].split("\n")[0]) correct = re.sub(r"^\d+\.\s*", "", raw).strip()[:80] pool = [w for w in WRONG_POOL if w.lower() not in correct.lower()] random.shuffle(pool) opts = pool[:3] + [correct] random.shuffle(opts) return correct, opts def start_quiz(ch_filter, state): status_dict = state.get("status", {}) deck = get_filtered(ch_filter, status_dict) random.shuffle(deck) state["qz_deck"] = [c["id"] for c in deck] state["qz_idx"] = 0 state["qz_score"] = 0 state["qz_answered"] = False return _render_qz(state) def _render_qz(state): deck_ids = state.get("qz_deck", [c["id"] for c in CARDS]) idx = state.get("qz_idx", 0) score = state.get("qz_score", 0) if idx >= len(deck_ids): total = len(deck_ids) pct = int(score / total * 100) if total else 0 ico = "🏆" if pct>=80 else "👍" if pct>=60 else "⚠️" if pct>=40 else "📚" msg = ("Excellent !" if pct>=80 else "Bien !" if pct>=60 else "Des lacunes — flashcards !" if pct>=40 else "Reprends tout !") q_md = f"# {ico} Score final : {pct}%\n\n**{score}/{total}** bonnes réponses\n\n*{msg}*" return (q_md, gr.update(choices=[], visible=False), gr.update(value="", visible=False), gr.update(visible=False, value="✅ Valider"), gr.update(visible=True), state) card = next(c for c in CARDS if c["id"] == deck_ids[idx]) correct, opts = _build_mcq(card) state["qz_correct"] = correct state["qz_answered"] = False state["qz_opts"] = opts q_md = (f"**{idx+1}/{len(deck_ids)}**  ·  {ch_badge(card['ch'])}  ·  " f"Score : **{score}/{idx}**\n\n---\n\n## ❓ {card['q']}") return (q_md, gr.update(choices=opts, value=None, visible=True, interactive=True), gr.update(value="", visible=False), gr.update(visible=True, value="✅ Valider"), gr.update(visible=False), state) def answer_qz(chosen, btn_label, state): if btn_label != "✅ Valider": state["qz_idx"] = state.get("qz_idx", 0) + 1 state["qz_answered"] = False return _render_qz(state) if not chosen or state.get("qz_answered"): return _render_qz(state) correct = state.get("qz_correct", "") is_ok = (chosen == correct) deck_ids = state.get("qz_deck", []) idx = state.get("qz_idx", 0) card_id = deck_ids[idx] if idx < len(deck_ids) else None card = next((c for c in CARDS if c["id"] == card_id), None) state["qz_answered"] = True if is_ok: state["qz_score"] = state.get("qz_score", 0) + 1 if card: state.setdefault("status", {})[str(card_id)] = "good" fb = f"### ✅ Bonne réponse !\n\n{card['a'] if card else ''}" else: if card: state.setdefault("status", {})[str(card_id)] = "bad" fb = f"### ❌ Incorrect\n\n**Bonne réponse :** {correct}\n\n---\n\n{card['a'] if card else ''}" deck_ids2 = state.get("qz_deck", []) idx2 = state.get("qz_idx", 0) score2 = state.get("qz_score", 0) card2 = next(c for c in CARDS if c["id"] == deck_ids2[idx2]) _, opts2 = _build_mcq(card2) q_md2 = (f"**{idx2+1}/{len(deck_ids2)}**  ·  {ch_badge(card2['ch'])}  ·  " f"Score : **{score2}/{idx2}**\n\n---\n\n## ❓ {card2['q']}") return (q_md2, gr.update(choices=opts2, value=chosen, interactive=False), gr.update(value=fb, visible=True), gr.update(visible=True, value="Question suivante →"), gr.update(visible=False), state) # ── Récap ── def build_recap(ch_filter): groups = { "Tout (76)": [("3","🔴 CH.3"), ("4","🔵 CH.4"), ("5","🟢 CH.5"), ("synth","🟡 SYNTHÈSE")], "CH.3 — Mobilité (19)": [("3","🔴 CH.3")], "CH.4 — Divergence (21)":[("4","🔵 CH.4")], "CH.5 — Convergence (20)":[("5","🟢 CH.5")], "Synthèse (5+)": [("synth","🟡 SYNTHÈSE")], } chapters = groups.get(ch_filter, groups["Tout (76)"]) md = "" for ch, label in chapters: md += f"\n---\n# {label}\n\n" for c in [x for x in CARDS if x["ch"] == ch]: md += f"**#{c['id']:02d} · {c['section']}**\n\n❓ *{c['q']}*\n\n✅ {c['a']}\n\n" return md # ── Carte de progression ── def build_map(state): sd = state.get("status", {}) md = "## 🗺 Carte de progression\n\n" for ch, label, ico in [("3","CH.3","🔴"),("4","CH.4","🔵"),("5","CH.5","🟢"),("synth","SYNTH","🟡")]: cards = [c for c in CARDS if c["ch"] == ch] g = sum(1 for c in cards if sd.get(str(c["id"]))=="good") o = sum(1 for c in cards if sd.get(str(c["id"]))=="ok") b = sum(1 for c in cards if sd.get(str(c["id"]))=="bad") bar = "█"*int(g/len(cards)*10) + "░"*(10-int(g/len(cards)*10)) md += f"{ico} **{label}** `[{bar}]` ✅{g} 🟡{o} ❌{b} / {len(cards)}\n\n" md += "\n---\n### Détail par carte\n\n" row = [] for c in CARDS: s = sd.get(str(c["id"])) ico2 = {"good":"✅","ok":"🟡","bad":"❌"}.get(s,"⬜") row.append(f"{ico2}`{c['id']:02d}`") if len(row) == 10: md += " ".join(row) + "\n\n"; row = [] if row: md += " ".join(row) + "\n\n" md += "\n**Légende :** ✅ Maîtrisé   🟡 Hésitant   ❌ À revoir   ⬜ Non vu" return md def reset_progress(state): state["status"] = {} return compute_stats({}), build_map(state), state # ══════════════════════════════════════════════════════════ # CSS # ══════════════════════════════════════════════════════════ CSS = """ @import url('https://fonts.googleapis.com/css2?family=Space+Mono:wght@400;700&family=Syne:wght@700;800&display=swap'); body, .gradio-container { font-family: 'Space Mono', monospace !important; } .app-header { background: linear-gradient(135deg,#111420,#181d2e); border:1px solid #252a3d; border-radius:14px; padding:28px 36px; margin-bottom:12px; } .app-header h1 { font-family:'Syne',sans-serif !important; font-size:2.4rem !important; font-weight:800 !important; letter-spacing:-2px; margin:0 !important; } .app-header h1 span { color:#e8440a; } .app-header p { color:#6b7090; font-size:.73rem; letter-spacing:2px; margin:6px 0 0; } """ # ══════════════════════════════════════════════════════════ # UI GRADIO # ══════════════════════════════════════════════════════════ with gr.Blocks(css=CSS, title="GéoRévision — Tectonique des Plaques", theme=gr.themes.Base( primary_hue="orange", neutral_hue="slate", font=[gr.themes.GoogleFont("Space Mono"), "monospace"] )) as demo: state = gr.State({}) gr.HTML("""

GÉORÉVISION

TECTONIQUE DES PLAQUES  ·  76 NOTIONS  ·  CH.3 · CH.4 · CH.5 · SYNTHÈSE  ·  MÉTHODE EBBINGHAUS + TESTING EFFECT

""") stats_md = gr.Markdown(compute_stats({})) with gr.Tabs(): # ═══ FLASHCARDS ═══ with gr.Tab("🃏 Flashcards"): with gr.Row(): fc_filter = gr.Dropdown(FILTER_CHOICES, value="Tout (76)", label="Filtre", scale=3) fc_start = gr.Button("▶ Démarrer", variant="primary", scale=1) fc_shuf = gr.Button("⟳ Mélanger", variant="secondary", scale=1) fc_prog = gr.Markdown("*Sélectionne un filtre et clique sur ▶ Démarrer*") with gr.Group(): fc_q = gr.Markdown("### ← Lance une session pour commencer !") fc_flip = gr.Button("👁 Voir la réponse", variant="primary") with gr.Group(): fc_a = gr.Markdown("") fc_s = gr.Markdown("") with gr.Row(visible=False) as fc_rating: fc_bad = gr.Button("❌ À revoir", variant="secondary", scale=1) fc_ok = gr.Button("🟡 Hésitant", variant="secondary", scale=1) fc_good = gr.Button("✅ Maîtrisé", variant="secondary", scale=1) with gr.Row(): fc_prev = gr.Button("← Précédent", variant="secondary", scale=1) fc_next = gr.Button("Suivant →", variant="secondary", scale=1) gr.Markdown("*💡 Méthode : lis la question, formule mentalement, puis révèle. Note honnêtement. Reviens sur le filtre **⚠ À revoir** régulièrement.*") FC = [fc_prog, fc_q, fc_a, fc_s, fc_flip, fc_rating, stats_md, state] fc_start.click(init_flashcard, [fc_filter, state], FC) fc_shuf.click(shuffle_fc, [state], FC) fc_flip.click(flip_fc, [state], FC) fc_prev.click(prev_fc, [state], FC) fc_next.click(next_fc, [state], FC) fc_bad.click( lambda s: rate_fc("bad", s), [state], FC) fc_ok.click( lambda s: rate_fc("ok", s), [state], FC) fc_good.click(lambda s: rate_fc("good", s), [state], FC) # ═══ QUIZ ═══ with gr.Tab("🎯 Quiz QCM"): with gr.Row(): qz_filt = gr.Dropdown(FILTER_CHOICES, value="Tout (76)", label="Filtre", scale=3) qz_start = gr.Button("▶ Lancer", variant="primary", scale=1) qz_q = gr.Markdown("### ← Lance le quiz pour commencer !") qz_opts = gr.Radio(choices=[], label="Ta réponse", visible=False) qz_fb = gr.Markdown("", visible=False) with gr.Row(): qz_val = gr.Button("✅ Valider", variant="primary", visible=False, scale=2) qz_restart = gr.Button("🔁 Recommencer", variant="secondary", visible=False, scale=1) QZ = [qz_q, qz_opts, qz_fb, qz_val, qz_restart, state] qz_start.click(start_quiz, [qz_filt, state], QZ) qz_val.click(answer_qz, [qz_opts, qz_val, state], QZ) qz_restart.click(start_quiz, [qz_filt, state], QZ) # ═══ RÉCAP ═══ with gr.Tab("📚 Récapitulatif"): recap_filt = gr.Dropdown( [c for c in FILTER_CHOICES if c != "⚠ À revoir"], value="Tout (76)", label="Chapitre" ) gr.Button("📖 Afficher", variant="primary").click( build_recap, [recap_filt], gr.Markdown("*Clique sur Afficher.*") ) recap_out = gr.Markdown("*Clique sur Afficher pour voir toutes les notions.*") gr.Button("📖 Afficher", variant="primary", visible=False) # Rebind correctement recap_filt.change(build_recap, [recap_filt], recap_out) # ═══ PROGRESSION ═══ with gr.Tab("🗺 Progression"): with gr.Row(): map_btn = gr.Button("🔄 Actualiser", variant="primary", scale=2) reset_btn = gr.Button("🗑 Réinitialiser", variant="stop", scale=1) map_out = gr.Markdown(build_map({})) map_btn.click(lambda s: build_map(s), [state], map_out) reset_btn.click(reset_progress, [state], [stats_md, map_out, state]) # ═══ AIDE ═══ with gr.Tab("ℹ️ Aide"): gr.Markdown(""" ## 🧠 Méthode scientifique | Principe | Source | Application | |----------|--------|-------------| | **Testing Effect** | Roediger & Karpicke, 2006 | Se tester > relire → chaque flashcard force une récupération active | | **Répétition espacée** | Ebbinghaus, 1885 | Filtre ⚠ À revoir → cible les lacunes au bon moment | | **Interleaving** | Kornell & Bjork, 2008 | Mode Tout (76) mélangé → améliore la discrimination | --- ## 🃏 Flashcards 1. Sélectionne un filtre → **▶ Démarrer** 2. Lis la question, formule **mentalement** ta réponse 3. Clique **👁 Voir la réponse** pour comparer 4. Note honnêtement : ❌ / 🟡 / ✅ 5. Reviens régulièrement sur **⚠ À revoir** ## 🎯 Quiz - 4 propositions par question - Feedback immédiat avec l'explication complète - Met à jour automatiquement la progression ## 📅 Plan de révision recommandé | Quand | Action | |-------|--------| | J-7 | Flashcards CH.3 complet | | J-5 | Flashcards CH.4 complet | | J-3 | Flashcards CH.5 + Synthèse | | J-2 | Quiz tout + ⚠ À revoir | | J-1 | Récap complet + Quiz final | | Jour J | Cartes ❌ uniquement | """) demo.load(lambda s: (compute_stats(s.get("status", {})), build_map(s)), [state], [stats_md, map_out]) if __name__ == "__main__": demo.launch()