<?xml version="1.0" encoding="UTF-8"?>
<quiz>
<question type="category">
  <category>
    <text>$course$/QCM de NSI/Première/JavaScript et DOM</text>
  </category>
  <info format="html">
    <text><![CDATA[<p>Bases de JavaScript pour interagir avec une page web :<br/>
intégration au HTML, syntaxe de base, manipulation du DOM<br/>
(Document Object Model), sélection et modification<br/>
d'éléments, exemples typiques.</p>]]></text>
  </info>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q01 : Rôle de JavaScript</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>À quoi sert principalement <strong>JavaScript</strong> dans<br/>
une page web ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Trois langages du Web côté navigateur : HTML<br/>
(structure), CSS (style), JavaScript<br/>
(comportement). JavaScript a aussi un<br/>
pendant côté serveur (Node.js).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>À mettre en forme les éléments</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>C'est le rôle de CSS.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>À envoyer des e-mails</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas son rôle (cela demande un<br/>
serveur).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>À rendre la page dynamique : interactions, animations, modification du contenu en réaction à l'utilisateur</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : HTML structure, CSS<br/>
embellit, JavaScript fait vivre. Trois<br/>
piliers complémentaires du Web.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>À structurer le contenu de la page</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>C'est le rôle de HTML.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q02 : Intégrer du JavaScript</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment <strong>inclure</strong> un fichier script.js dans<br/>
une page HTML ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Variantes : script externe (src), script<br/>
inline (&lt;script&gt;...&lt;/script&gt;), module<br/>
(type="module"). Toujours préférer le<br/>
script externe pour la maintenance.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>html<br/>
&lt;link href="script.js"&gt;<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : &lt;link&gt; est pour les feuilles<br/>
de style.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>`<code>html<br/>
&lt;script src="script.js"&gt;&lt;/script&gt;<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : balise &lt;script&gt; avec<br/>
src. Préférable de placer en bas du<br/>
&lt;body&gt; (ou avec defer dans &lt;head&gt;)<br/>
pour ne pas bloquer le rendu.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>html<br/>
&lt;import "script.js"&gt;<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : balise inexistante en HTML.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>html<br/>
&lt;js src="script.js"&gt;&lt;/js&gt;<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : balise inexistante.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q03 : Déclaration de variable</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quel mot-clé moderne déclare une variable<br/>
<strong>modifiable</strong> en JavaScript ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Trois mots-clés : var (ancien, à éviter),<br/>
let (modifiable), const (constante,<br/>
non réassignable). Privilégier const par<br/>
défaut, let quand on doit modifier.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>var</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Possible mais désuet (portée<br/>
déroutante). Préférer let.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>def</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est Python.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>let</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : let x = 5; déclare une<br/>
variable de portée bloc, modifiable.<br/>
Standard depuis ES6 (2015).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>int</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : JS n'est pas typé statiquement.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q04 : Afficher dans la console</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment <strong>afficher</strong> un message dans la console<br/>
du navigateur en JavaScript ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Variantes : console.warn, console.error,<br/>
console.table (pour les objets/tableaux),<br/>
console.dir (pour les éléments DOM).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>console.log("Hello")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : console.log est le<br/>
standard. Visible dans l'onglet<br/>
« Console » des outils développeur (F12).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>echo "Hello"</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est PHP/Bash.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>print("Hello")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est Python.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>puts "Hello"</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est Ruby.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q05 : DOM</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Que signifie <strong>DOM</strong> ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Le DOM est la « porte d'entrée » de<br/>
JavaScript vers la page. Chaque élément HTML<br/>
devient un objet manipulable.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Dynamic Output Module</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Aucune spécification ne<br/>
porte ce nom. Le DOM est<br/>
la représentation arborescente<br/>
d'un document HTML, exposée<br/>
aux scripts pour permettre<br/>
la modification dynamique<br/>
de la page.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Display Order Manager</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Cette dénomination est<br/>
inventée. Le sigle DOM<br/>
renvoie au <em>Document<br/>
Object Model</em>, normalisé<br/>
par le W3C dans le cadre<br/>
de la modélisation des<br/>
documents HTML.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Document Object Model</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : représentation<br/>
arborescente de la page HTML, accessible<br/>
et modifiable depuis JavaScript via des<br/>
objets et des méthodes.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Domain Object Method</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Cette appellation n'est<br/>
pas standard. Le sigle<br/>
DOM, défini par le W3C,<br/>
signifie <em>Document Object<br/>
Model</em> : il décrit la<br/>
structure d'un document<br/>
HTML sous forme d'arbre<br/>
d'objets.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q06 : Sélection par id</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment récupérer en JavaScript l'élément ayant<br/>
id="titre" ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Alternative moderne et flexible :<br/>
document.querySelector("#titre") (accepte<br/>
tout sélecteur CSS).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>getElementById(titre)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : il manque document. et les<br/>
guillemets autour de la chaîne.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>document.find("#titre")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas en JavaScript natif (oui en<br/>
jQuery).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>document.getId("titre")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : méthode inexistante.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>document.getElementById("titre")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : méthode classique. Renvoie<br/>
l'objet DOM correspondant ou null si<br/>
l'id n'existe pas.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q07 : Modifier le texte</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment changer le <strong>texte</strong> d'un élément el<br/>
sélectionné par DOM ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>textContent (texte pur) vs innerHTML<br/>
(HTML brut). Préférer textContent quand on<br/>
affiche du contenu utilisateur (sécurité).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>el.value = "..."</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : value concerne les champs de<br/>
formulaire (&lt;input&gt;, &lt;select&gt;).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>el.text = "..."</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : propriété inexistante.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>el.textContent = "..."</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : textContent modifie le<br/>
contenu textuel. Pour insérer du HTML<br/>
(avec balises), utiliser innerHTML<br/>
(mais attention aux failles XSS).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>el.changeText("...")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : méthode inexistante.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q08 : DOM comme arbre</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle structure de données est utilisée pour<br/>
modéliser le DOM ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Vocabulaire : nœud parent, nœud enfant,<br/>
frère, racine, feuille. Chaque élément du<br/>
DOM est un <strong>nœud</strong>, qui peut être de type<br/>
« élément », « texte », « attribut », etc.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Un graphe orienté avec cycles</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : un arbre n'a pas de cycles.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Un arbre dont la racine est &lt;html&gt;</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : structure hiérarchique<br/>
reflétant l'imbrication des balises HTML.<br/>
Chaque élément a un parent et peut avoir<br/>
des enfants.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Une liste plate</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : trop limité.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Un dictionnaire</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas la structure adaptée.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q09 : Réagir à un clic</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment exécuter une fonction f quand on<br/>
clique sur un bouton btn ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>addEventListener est la méthode standard.<br/>
Premier paramètre = type d'événement<br/>
("click", "input", "submit"...),<br/>
second = fonction à exécuter.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>btn.onClick = f</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Casse incorrecte. C'est onclick (tout<br/>
en minuscules).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>btn.click = f</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : .click est une méthode (pour<br/>
déclencher le clic), pas un setter de<br/>
handler.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>btn.addEventListener("click", f)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : méthode moderne et<br/>
recommandée. Permet d'ajouter plusieurs<br/>
écouteurs sans s'écraser mutuellement.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>btn.listen(f)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : méthode inexistante en JS natif.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q10 : Trace simple</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Que fait ce code ?<br/>
`<code>javascript<br/>
let x = 5;<br/>
x = x + 3;<br/>
console.log(x);<br/>
</code>`</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Attention au piège de l'addition vs<br/>
concaténation : 5 + "3" = "53" en JS<br/>
(coercition de type). C'est une source<br/>
classique de bugs.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Erreur</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : code valide.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Affiche 53 (concaténation)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : 3 est un nombre, pas une chaîne,<br/>
donc addition arithmétique.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Affiche 5</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : on additionne 3.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Affiche 8</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : x devient 5 + 3 = 8, puis<br/>
on l'affiche.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q11 : querySelector</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Que renvoie<br/>
document.querySelector(".bouton") ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>querySelector(sel) = premier élément ;<br/>
querySelectorAll(sel) = NodeList. Plus<br/>
flexibles que getElementById /<br/>
getElementsByClassName.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Le bouton cliqué le plus récemment</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>La méthode querySelector<br/>
interroge la structure du<br/>
document, pas son historique<br/>
d'interaction. Le suivi de<br/>
l'élément actif passe par<br/>
document.activeElement ou<br/>
des gestionnaires d'évènements.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Une chaîne contenant le HTML du bouton</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : renvoie l'objet DOM, pas une<br/>
chaîne.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Le premier élément ayant la classe « bouton » (ou null si aucun)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : querySelector accepte<br/>
un sélecteur CSS (classe, id,<br/>
combinaisons...) et renvoie le premier<br/>
élément correspondant.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Tous les éléments ayant la classe « bouton »</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est querySelectorAll qui<br/>
renvoie tous les éléments.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q12 : textContent vs innerHTML</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle est la différence entre textContent et<br/>
innerHTML ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Sécurité : si le contenu vient de<br/>
l'utilisateur, <strong>toujours</strong> utiliser<br/>
textContent pour éviter les attaques XSS<br/>
(injection de scripts).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Aucune</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : différence importante.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>textContent traite la valeur comme du texte brut (échappant les balises) ; innerHTML interprète le HTML</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : el.textContent =<br/>
"&lt;b&gt;x&lt;/b&gt;" affiche littéralement<br/>
&lt;b&gt;x&lt;/b&gt; ; el.innerHTML = "&lt;b&gt;x&lt;/b&gt;"<br/>
affiche x en gras.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>textContent n'existe pas</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : méthode standard.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>innerHTML est plus rapide</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : tendance inverse, et la sécurité<br/>
prime.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q13 : Conditions</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle syntaxe est <strong>valide</strong> en JavaScript pour<br/>
une condition ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Syntaxe similaire à C/Java. Astuce : les<br/>
points-virgules sont techniquement<br/>
facultatifs (insertion automatique), mais<br/>
recommandés pour éviter les pièges.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
if (x &gt; 0) {<br/>
    console.log("positif");<br/>
}<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : parenthèses autour de la<br/>
condition, accolades pour le bloc, points-<br/>
virgules en fin d'instruction.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
if x &gt; 0 then<br/>
    console.log("positif")<br/>
end<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est la syntaxe Lua/Ruby.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
if x &gt; 0:<br/>
    console.log("positif")<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est la syntaxe Python.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
if x &gt; 0:<br/>
    console.log "positif"<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : syntaxe inventée.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q14 : Boucle for</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle syntaxe for JavaScript itère <strong>5 fois</strong> ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Variantes modernes : for...of (itérer sur<br/>
les valeurs d'un tableau), for...in<br/>
(itérer sur les clés d'un objet),<br/>
forEach (méthode des tableaux).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
for i in range(5)<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est Python.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
for i = 0 to 5<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est BASIC/Pascal.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
loop 5 times<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : syntaxe inventée.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
for (let i = 0; i &lt; 5; i++) { ... }<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : trois parties dans le<br/>
for : initialisation, condition,<br/>
incrément (séparées par ;).</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q15 : Définir une fonction</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle est la syntaxe valide d'une fonction JS ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Syntaxe moderne (« arrow function ») :<br/>
const carre = x =&gt; x * x;. Plus concise<br/>
mais subtilités sur this.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
def carre(x): return x * x<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est Python.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
method carre(x) { ... }<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas de mot-clé method global<br/>
en JS.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
carre(x) -&gt; x * x<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : syntaxe inventée.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
function carre(x) { return x * x; }<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : mot-clé function,<br/>
accolades pour le bloc, return pour la<br/>
valeur.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q16 : Modifier un attribut</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment changer l'<strong>attribut src</strong> d'une image ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Pour les attributs standards (src, href,<br/>
alt, id...), on accède directement par<br/>
la propriété. Pour les attributs<br/>
personnalisés ou spéciaux : setAttribute/<br/>
getAttribute.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>img.attribute("src", "...")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : méthode inexistante.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>img.changeSrc("...")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : méthode inexistante.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>img.src = "nouvelle.jpg"</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : on peut accéder aux<br/>
attributs comme à des propriétés JS.<br/>
Forme alternative :<br/>
img.setAttribute("src", "nouvelle.jpg").</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>img.attr.src = "..."</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : .attr n'existe pas.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q17 : Créer un élément</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment créer dynamiquement un nouveau<br/>
paragraphe &lt;p&gt; et l'<strong>ajouter</strong> au &lt;body&gt; ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Variantes pour ajouter : appendChild (à la<br/>
fin), prepend (au début), insertBefore<br/>
(à un endroit précis).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
add("p", "Hello");<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : fonction inexistante.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
new HTMLElement("p");<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas la bonne API.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
document.body.innerHTML = "&lt;p&gt;Hello&lt;/p&gt;";<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Cela écrase <strong>tout</strong> le contenu de<br/>
&lt;body&gt;. Pas équivalent (et peu sûr).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>`<code>javascript<br/>
const p = document.createElement("p");<br/>
p.textContent = "Hello";<br/>
document.body.appendChild(p);<br/>
</code>`</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : trois étapes, à savoir<br/>
créer, remplir, ajouter au DOM.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q18 : == vs ===</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle est la différence entre == et === en<br/>
JavaScript ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Règle d'or : <strong>toujours utiliser ===</strong> sauf<br/>
cas très précis. Idem pour l'inégalité :<br/>
!== plutôt que !=.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Aucune</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : différence importante !</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>== compare avec conversion de type (5 == "5" est true) ; === compare strictement (sans conversion : 5 === "5" est false)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : préférer === pour<br/>
éviter les surprises de coercition.<br/>
Source de bugs subtils en JS.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>== n'existe pas en JS moderne</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : il existe, mais à éviter.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>=== est obsolète</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est la version recommandée.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q19 : Tableau JavaScript</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment déclarer un <strong>tableau</strong> de trois<br/>
éléments en JS ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Tableaux JS : indexés à partir de 0,<br/>
hétérogènes (peuvent contenir n'importe quel<br/>
type). Très polyvalents.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>let t = [1, 2, 3];</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : crochets, virgules.<br/>
Accès : t[0] (= 1). Méthodes :<br/>
t.push, t.pop, t.length...</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Array(1, 2, 3)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Forme valide mais on préfère la<br/>
notation littérale [1, 2, 3].</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>let t = {1, 2, 3};</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : {} désigne un objet en JS,<br/>
avec syntaxe {cle: valeur}.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>let t = (1, 2, 3);</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : ce serait un tuple Python.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q20 : Événement et données</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Comment savoir <strong>sur quel élément</strong> un<br/>
événement a eu lieu dans le gestionnaire ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>L'objet event est passé automatiquement<br/>
au callback. Propriétés utiles :<br/>
event.target (qui a déclenché),<br/>
event.type (type d'événement),<br/>
event.preventDefault() (annuler le<br/>
comportement par défaut).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>En relançant le code</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Relancer le script ne<br/>
fournit aucune information<br/>
supplémentaire. C'est l'objet<br/>
event, automatiquement<br/>
transmis au gestionnaire,<br/>
qui contient toutes les<br/>
données utiles.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Impossible</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : il existe une variable<br/>
d'événement.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Avec une variable globale</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Pratique fragile, à éviter.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Via event.target (passé au callback)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse :<br/>
btn.addEventListener("click", (e) =&gt;<br/>
console.log(e.target)); affiche<br/>
l'élément qui a déclenché l'événement.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q21 : Débogage</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Pour déboguer un script JavaScript dans le<br/>
navigateur, quel outil utiliser ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Astuces : debugger; pour mettre un point<br/>
d'arrêt en code ; console.table(arr) pour<br/>
visualiser un tableau ; console.dir(elem)<br/>
pour explorer un nœud DOM.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Visual Studio</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>IDE pour développer, pas pour déboguer<br/>
dans le navigateur.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Notepad</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Pas un débogueur.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Un script Python</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Python est un langage<br/>
autonome, exécuté côté<br/>
serveur ou en local. Le<br/>
débogage d'un script<br/>
JavaScript se fait dans<br/>
le navigateur, là où le<br/>
code s'exécute réellement.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Les outils développeur du navigateur (touche F12) : console, débogueur, inspecteur, onglet Network</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : Chrome DevTools, Firefox<br/>
DevTools, Safari Web Inspector. Tous<br/>
gratuits et puissants. Apprendre à les<br/>
utiliser est essentiel.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q22 : DOM pas encore prêt</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Mon script JS fait document.getElementById<br/>
mais renvoie null. Pourquoi ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Trois solutions : (1) placer le &lt;script&gt; à<br/>
la fin de &lt;body&gt;, (2) &lt;script defer&gt;<br/>
dans &lt;head&gt;, (3) document.addEventListener<br/>
("DOMContentLoaded", ...).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>L'élément n'existe pas dans la page</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Possible aussi, mais la cause la plus<br/>
fréquente est le timing.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>JavaScript ne peut pas lire le DOM</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est exactement son rôle.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Le script s'exécute avant que le DOM ait été parsé. L'élément n'existe pas encore.</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : si &lt;script&gt; est dans<br/>
&lt;head&gt; sans defer, il s'exécute avant<br/>
le &lt;body&gt;. Solutions : placer le<br/>
script en bas du &lt;body&gt;, utiliser<br/>
defer, ou encapsuler dans un<br/>
écouteur DOMContentLoaded.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Le navigateur est buggé</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Très improbable.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q23 : Coercition de type</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Que renvoie "5" + 3 en JavaScript ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Comparaison : "5" - 3 = 2 (le - force<br/>
la conversion en nombre). Imprévisible :<br/>
toujours convertir explicitement avec<br/>
Number(s) ou parseInt(s).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>"53" (chaîne, par concaténation)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : avec +, si l'un des<br/>
opérandes est une chaîne, l'autre est<br/>
converti en chaîne et on concatène.<br/>
Comportement spécifique à JS, source<br/>
fréquente de bugs.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Erreur de type</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : JS coercite silencieusement.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>5</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : 5 ignorerait le 3.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>8</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas d'addition arithmétique car<br/>
un opérande est une chaîne.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q24 : Sécurité innerHTML</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Pourquoi el.innerHTML = userInput; est-il<br/>
<strong>dangereux</strong> ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Règle d'or : ne jamais insérer de contenu<br/>
utilisateur via innerHTML. Préférer<br/>
textContent, ou un nettoyage avec une<br/>
bibliothèque (DOMPurify).</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Parce que c'est lent</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas d'impact significatif sur la<br/>
performance.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Parce qu'un utilisateur malveillant peut injecter du code HTML/JS (attaque XSS : Cross-Site Scripting)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : si userInput contient<br/>
&lt;script&gt;...&lt;/script&gt;, le code malveillant<br/>
s'exécute dans le navigateur. Solution :<br/>
utiliser textContent (qui échappe) ou<br/>
assainir l'entrée.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Parce que innerHTML est obsolète</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : pas obsolète.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Aucun risque réel</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : XSS est l'une des<br/>
vulnérabilités les plus exploitées.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q25 : Synthèse</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Parmi les affirmations suivantes sur JavaScript<br/>
et le DOM, laquelle est <strong>fausse</strong> ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Mnémonique : « toujours triple égal » en<br/>
JS, sauf cas très exceptionnels documentés.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>addEventListener permet d'ajouter plusieurs gestionnaires d'événements à un élément</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Vrai : avantage majeur sur onclick =.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>document.getElementById(id) renvoie le premier élément ou null</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Vrai : comportement standard.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>== et === sont synonymes en JavaScript</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Faux (donc bonne réponse) : == fait<br/>
une comparaison <strong>avec coercition</strong> (donc<br/>
parfois surprenante : 0 == false,<br/>
"" == 0...) ; === est une<br/>
comparaison <strong>stricte</strong>, sans conversion<br/>
de type.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>Le DOM est une représentation arborescente de la page HTML</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Vrai : structure clé du Web.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q26 : let, const et var</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Lequel de ces mots-clés permet, en JavaScript<br/>
moderne, de déclarer une variable dont la valeur ne<br/>
pourra <strong>plus changer</strong> après l'affectation initiale ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Recommandation moderne : préférer const par<br/>
défaut, et n'utiliser let que si on a vraiment<br/>
besoin de réaffecter la variable. Le mot-clé var<br/>
est conservé pour la rétrocompatibilité mais ne<br/>
doit plus apparaître dans du code récent.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>var</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : var autorise au contraire la<br/>
réaffectation. C'est l'ancien mot-clé de<br/>
déclaration, à éviter aujourd'hui car il a<br/>
également une portée moins prévisible.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>function</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : function sert à déclarer une<br/>
fonction, pas une variable. Il n'a aucun<br/>
rapport avec la modification d'une valeur.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>const</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : const interdit la<br/>
réaffectation après l'initialisation. Cela<br/>
n'empêche cependant pas de <strong>muter</strong> un objet<br/>
ou un tableau (modifier ses propriétés ou ses<br/>
éléments).</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>let</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : let déclare une variable dont la<br/>
valeur peut être réaffectée. Pour interdire la<br/>
réaffectation, on utilise plutôt const.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q27 : Modifier les classes d'un élément</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Pour ajouter la classe actif à un élément el<br/>
sélectionné via querySelector, quelle instruction<br/>
utiliser en JavaScript moderne ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Avantages de classList : on n'écrase pas les<br/>
autres classes, le code est lisible, et<br/>
toggle("actif") est un raccourci pratique pour<br/>
ajouter ou retirer la classe selon son état<br/>
courant. C'est l'API à privilégier pour basculer<br/>
entre des états visuels.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>el.style.actif = true</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : la propriété style permet de<br/>
modifier les styles <strong>inline</strong>, pas la liste<br/>
des classes. Il n'existe pas de style<br/>
actif.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>el.addClass("actif")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : addClass n'est pas une méthode<br/>
native du DOM. C'est une fonction de jQuery,<br/>
librairie qui n'est pas disponible par défaut<br/>
dans le navigateur.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>el.classList.add("actif")</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : la propriété classList<br/>
fournit une API dédiée à la gestion des<br/>
classes : add, remove, toggle,<br/>
contains. C'est la méthode recommandée<br/>
aujourd'hui.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>`el.class = "actif"</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : la propriété DOM s'appelle<br/>
className, pas class (qui est un mot<br/>
réservé en JavaScript). Et l'assigner écrase<br/>
toutes les classes existantes au lieu d'en<br/>
ajouter une.</p>]]></text>
    </feedback>
  </answer>
</question>

<question type="multichoice">
  <name>
    <text>JavaScript et DOM — Q28 : getElementById vs querySelector</text>
  </name>
  <questiontext format="html">
    <text><![CDATA[<p>Quelle est la différence entre document.getElementById("titre")<br/>
et document.querySelector("#titre") ?</p>]]></text>
  </questiontext>
  <generalfeedback format="html">
    <text><![CDATA[<p>Bonne pratique : utiliser getElementById quand<br/>
on cible précisément un identifiant ;<br/>
querySelector lorsqu'on a besoin d'un sélecteur<br/>
complexe (par exemple nav &gt; a.actif:first-child).<br/>
Pour récupérer plusieurs éléments,<br/>
querySelectorAll renvoie une NodeList<br/>
itérable.</p>]]></text>
  </generalfeedback>
  <defaultgrade>1.0</defaultgrade>
  <penalty>0.0</penalty>
  <hidden>0</hidden>
  <single>true</single>
  <shuffleanswers>true</shuffleanswers>
  <answernumbering>abc</answernumbering>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>La première renvoie une liste, la seconde un seul élément</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : getElementById renvoie un seul<br/>
élément (ou null). C'est getElementsByClassName<br/>
ou querySelectorAll qui renvoient une<br/>
collection.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>querySelector ne fonctionne que sur les pages chargées par AJAX</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : querySelector fonctionne sur tout<br/>
élément du DOM, quel que soit son mode de<br/>
chargement.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="100" format="html">
    <text><![CDATA[<p>Aucune dans le résultat ; les deux retournent l'élément dont l'id vaut titre (s'il existe)</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Bonne réponse : pour un identifiant donné, les<br/>
deux méthodes renvoient le même élément.<br/>
getElementById est plus rapide et plus<br/>
ancienne ; querySelector accepte tout<br/>
sélecteur CSS, ce qui la rend plus polyvalente<br/>
mais légèrement plus lente.</p>]]></text>
    </feedback>
  </answer>
  <answer fraction="0" format="html">
    <text><![CDATA[<p>La première fonctionne sur les classes, la seconde sur les identifiants</p>]]></text>
    <feedback format="html">
      <text><![CDATA[<p>Erreur : c'est l'inverse. getElementById<br/>
cible un identifiant ; querySelector accepte<br/>
tout sélecteur CSS, dont les classes<br/>
(.maClasse).</p>]]></text>
    </feedback>
  </answer>
</question>

</quiz>
