legacy/outils/psylvia/Psylvia.js

// chargement de la feuille de style
import './psylvia.css'
import { j3pShowError } from 'src/legacy/core/functions'
import ALrepere from 'src/legacy/outils/psylvia/ALrepere'
import Tortue from 'src/legacy/outils/psylvia/Tortue'
import { afficher, filmDessinEfface, filmDessinGetElt, filmDessinSetElt, filmTortueEfface, filmTortueGetElt, filmTortueSetElt, getFreqTortue, papGetIndex, papInc, papReset, setFreqTortue } from './functions'

const w = window
const isFirefox = (navigator.userAgent.indexOf('Firefox') > -1)
const ALisIE = w.ActiveXObject

// recherche du num de la ligne de l’erreur
w.onerror = function onError (message, url, linenumber) {
  // pourquoi linenumber -1 ??
  souligne(numLignes[linenumber - 1], 'erreur')
}
// des trucs qui doivent être globaux (utilisé dans du code en string)
// décimales
w.ALprecision = 6
/** @type {ALrepere} */
w.ALrep = null
/** @type {Tortue} */
w.ALtortue = null

w.ALfcts = {}
w.ALvarAnciennes = {}

// tableau d’instructions formatees
w.ALformatees = []
// pour la gestion du pas à pas
// film de l’exécution du programme utilisateur
w.ALstop = false
w.AL1clic = undefined
w.ALtortueActive = false
// ralenti tortue
w.ALnb = undefined
w.ALangle = undefined

// nom du programme écrit par utilisateur
w.ALcle = ''
// pour le dessin
w.ALfenetre = { xMin: -5, yMin: -5, xMax: 5, yMax: 5, stroke: 'noir', fill: 'red' }

// fcts globales plus loin, après nos fonctions locales

// variables globales aux fcts de ce module
let actionFusee = () => undefined
let clic = false
/** @type {string} */
let controles
let dessinActif = false
let index = -1
// index du pas à pas
let indexPaP
// film de l’exécution du programme utilisateur (pas à pas)
let film = []
let limite = 5000
let vars = {}
let numLignes = []
let texteAffiche = ''
// liste nom: string
let verifs = {}
// liste nom: boolean
const isVerifOk = {}

function chercheElement (conteneur, tag, index) {
  tag = tag.toUpperCase()
  const c = document.getElementById(conteneur).childNodes
  let i = -1
  let j = -1
  while (i < c.length && j < index) {
    i++
    if (c?.[i]?.tagName?.toUpperCase() === tag) { j++ }
  }
  if (i < c.length) { return c[i] }
  return false
}

function compte () {
  index++
  // on propose l’arrêt du programme
  if (index >= limite) {
    let noLimit = false
    if (index === limite) {
      // eslint-disable-next-line no-alert
      noLimit = confirm('Les ' + limite + ' étapes sont dépassées.\nBoucle sans fin possible.\nFaut-il continuer ?')
    }
    if (!noLimit) {
      throw Error('Programme interrompu à votre demande.')
    }
    // sinon on augmente la limite
    limite += Math.min(20 * limite, 100000)
  }
}

function ecritSource (texte) {
  const div = document.getElementById('ALgauche')
  // nettoyage et récup de l’element pre
  let pre = div.firstChild
  while (pre) {
    div.removeChild(pre)
    pre = div.firstChild
  }
  pre = document.createElement('pre')
  texte = texte.replace(/\/\/br/g, '<br>')
  texte = texte.replace(/&lt;sup&gt;/gi, '<sup>')
  texte = texte.replace(/&lt;\/sup&gt/gi, '</sup>')
  pre.innerHTML = texte
  div.appendChild(pre)
}

function exec (prog) {
  limite = 5000
  index = -1
  const err777 = new Error()
  try {
  // eslint-disable-next-line no-eval
    eval(prog)
  } catch (err) {
    if (w.ALstop) {
      afficher('Programme arrêté par l’utilisateur')
    } else {
      afficher('ERREUR : ', err.message)
      // pour encadrer en rouge si pas firefox
      if (isFirefox) {
        const numLigne = numLignes[err.lineNumber - err777.lineNumber - 2]
        souligne(numLigne, 'erreur')
      } else {
        // eslint-disable-next-line no-eval
        eval(prog)
      }
    }
  }
  document.getElementById('ALdroite').appendChild(document.createElement('hr'))
  // contrôle des variables à vérifier
  for (const nom in verifs) {
    isVerifOk[nom] = (String(verifs[nom]) === String(w[nom]))
    if (String(verifs[nom]) !== String(w[nom])) {
      afficher('La variable ' + nom + ' a une valeur ' + w[nom] + ' incorrecte')
    }
  }
  // destruction des variables utilisateur
  for (const i in vars) delete w[i]
}

function faitFichier (type, chaine) {
  const json = {}
  if (arguments.length === 0) { type = 'algo' }
  switch (type) {
    case 'algo':
      var donnee = litSource()
      donnee = special(donnee)
      var t = donnee.split('//br;')
      if (t[0] == 'début_algo') { t.shift() }
      if (t[t.length - 1] == 'fin_algo') { t.pop() }
      if (t[t.length - 1].indexOf('contrôler') > -1) { t.pop() }
      donnee = t.join('//br;')
      json.algo = donnee
      if (controles) {
        try {
          json.controles = JSON.parse(controles)
        } catch (e) {
          console.error(e)
          afficher('ligne controlées non valide :\n' + controles + '\n', e)
        }
      }
      break
    case 'repas':
      json.repas = chaine
      break
    case 'itineraire':
      json.itineraire = chaine
      break
  }
  const lien = document.getElementById('ALlien')
  if (lien) {
    if (!ALisIE) {
      lien.href = 'data:;charset=utf8;,' + encodeURIComponent(JSON.stringify(json))
    } else {
      const ifr = document.getElementById('ALiframe')
      const ifrDoc = ifr.contentWindow.document
      ifrDoc.open('text/plain', 'replace')
      ifrDoc.charset = 'utf-8'
      ifrDoc.write(JSON.stringify(json))
      ifrDoc.close()
      document.charset = 'utf-8'
      ifrDoc.lien.href = 'javascript:ALsauveIE()'
    }
  }
}

// reglage du nombre de décimales des sorties
function faitPrecision () {
  const entree = document.getElementById('ALinputPrecis')
  let n = parseInt(entree.value)
  if (isNaN(n) || n < 0) {
    n = 6
    entree.value = '6'
  }
  w.ALprecision = n
}

function initDemos () {
  const s = document.getElementById('ALselectDemo')
  let i = 0
  let opt
  let h1 = chercheElement('ALdemos', 'H1', 0)
  while (h1) {
    opt = document.createElement('option')
    opt.innerHTML = h1.innerHTML
    s.appendChild(opt)
    i++
    h1 = chercheElement('ALdemos', 'h1', i)
  }
}

function initialiseTortue () {
  const droite = document.getElementById('ALdroiteTortue')
  if (!w.ALtortue) {
    w.ALtortue = new Tortue(droite.offsetWidth - 20, droite.offsetHeight - 20)
  } else {
    w.ALtortue.reInitialise()
  }
}

function lancer (preserveIndexPasApas) {
  voirDiv('ALaide', false)
  if (!preserveIndexPasApas) {
    preserveIndexPasApas = false
  }
  w.ALstop = false
  const prog = parseAlgo(preserveIndexPasApas)
  if (dessinActif) w.ALrep = new ALrepere()
  exec(prog)
}

function litSource (avecLignes) {
  const h = document.getElementById('ALgauche')
  let reg, s
  const expo = ['⁰', '¹', '²', '³', '⁴', '⁵', '⁶', '⁷', '⁸', '⁹']
  // suppression des balises pre surnuméraires des navigateurs webkit
  let pre = h.firstChild
  let texte = ''
  if (pre.innerHTML) { texte = pre.innerHTML }
  while (pre.nextSibling) {
    pre = pre.nextSibling
    if (pre.innerHTML) {
      if (texte != '') { texte += '<br>' }
      texte += pre.innerHTML
    }
  }
  texte = '<pre>' + texte + '</pre>'
  texte = texte.replace(/<sup>/gi, '&lt;sup&gt;')
  texte = texte.replace(/<\/sup>/gi, '&lt;/sup&gt;')
  for (let i = 0; i < expo.length; i++) {
    reg = new RegExp(expo[i], 'gi')
    s = '&lt;sup&gt;' + i + '&lt;/sup&gt;'
    texte = texte.replace(reg, s)
  }
  texte = texte.replace(/\n/g, '//br;')
  texte = texte.replace(/<br>/gi, '//br;')
  texte = texte.replace(/\n/g, '//br;')
  texte = texte.replace(/→/gi, ' → ')

  const div = document.createElement('div')
  div.innerHTML = texte
  texte = div.textContent
  if (avecLignes) {
    texte = texte.replace(/\/\/br/gi, '\n')
  }
  return texte
}

function metronome () {
  const freq = document.getElementById('ALfreq')
  freq.parentNode.style.display = 'none'
  setFreqTortue(25 * parseInt(freq.value))
  w.ALtimer = setInterval(pasApas, getFreqTortue() * 40)
}

function modeTortue () {
  // function redim (grand) {
  //   let rg, rt, rb, ld
  //   const t = ['ALdroite', 'ALaide', 'ALdroiteTortue']
  //   if (grand) {
  //     rg = '60%'
  //     rt = '62%'
  //     rb = '61%'
  //     ld = '41%'
  //   } else {
  //     rg = '51%'
  //     rt = '57%'
  //     rb = '53%'
  //     ld = '50%'
  //   }
  //   let e = document.getElementById('ALgauche')
  //   e.style.right = rg
  //   e = document.getElementById('ALtouches')
  //   e.style.right = rt
  //   e = document.getElementById('ALboutons')
  //   e.style.right = rb
  //   for (let i = 0; i < t.length; i++) {
  //     e = document.getElementById(t[i])
  //     e.style.left = ld
  //   }
  // }
  w.ALtortueActive = !w.ALtortueActive
  const d = document.getElementById('ALdroiteTortue')
  const r = document.getElementById('ALrepasTortue')
  if (w.ALtortueActive) {
    // redim(true);
    d.style.visibility = 'visible'
    r.style.display = 'block'
    initialiseTortue()
  } else {
    // redim(false);
    r.style.display = 'none'
    d.style.visibility = 'hidden'
  }
}

// traduction de l’algo en javascript
function parseAlgo (preservePasApas) {
  texteAffiche = ''

  function gras (s) {
    return '££b' + s + '££c'
  }

  function traduit (s) {
    function op (c) {
      const t = ['+', '-', '/', '*', '%', '<', '>', '=', '≤', '≥', '≠']
      for (let i = 0; i < t.length; i++) {
        if (c == t[i]) { return true }
      }
      return false
      // return c == '+' || c == '-' || c == '*' || c == '/' || c == '%'
    }
    // recherche des puissances
    function parsePuissance (s) {
      let j = s.lastIndexOf('</sup>')
      if (j == -1) { return s }
      // ce qui est avant la balise
      let s1 = s.substring(0, j)
      // on traite récusrivement les autres exposants :
      // il ne restera donc plus que la balise <sup> ouvrante
      s1 = parsePuissance(s1)
      // ce qui est après le dernier </sup>
      const s2 = s.substring(j + 6)
      // recherche de l’exposant
      j = s1.lastIndexOf('<sup>')
      const exposant = s1.substring(j + 5)
      s1 = s1.substring(0, j)
      // recherche de la mantisse
      let k = s1.length - 1; let debutM = -1
      const parenth = (s1.charAt(k) == ')')
      let finM
      if (parenth) { finM = k } else { finM = k + 1 }
      let nivPar = 0
      while (debutM == -1 && k >= 0) {
        if (s1.charAt(k) == ')') { nivPar++ }
        if (s1.charAt(k) == '(') { nivPar-- }
        k--
        if ((((nivPar < 0 || parenth) && nivPar == 0) || op(s1.charAt(k))) && !parenth) {
          debutM = k
        }
      }
      debutM++
      if (nivPar < 0) debutM++
      if (parenth) {
        return s1.substring(0, debutM) + 'Math.pow(' + s1.substring(debutM + 1, finM) + ',' + exposant + ')' + s2
      }
      return s1.substring(0, debutM) + 'Math.pow(' + s1.substring(debutM, finM) + ',' + exposant + ')' + s2
    }

    // transformer x<sup>y</sup> en Math.pow(x,y)
    s = parsePuissance(s)
    // opérateurs logiques
    if (s == 'ou') { return '||' }
    if (s == 'et') { return '&&' }
    // autres
    let reg
    const t1 = ['racine', 'vrai', 'faux', '=', '≤', '≥', '≠', 'π', 'faire', 'alors', "'"]
    const t2 = ['Math.sqrt', 'true', 'false', '==', '<=', '>=', '!=', 'Math.PI', '', '', "\\'"]
    for (let i = 0; i < t1.length; i++) {
      reg = new RegExp(t1[i], 'gi')
      s = s.replace(reg, t2[i])
    }
    return s
  }

  // cherche mot1 suivi eventuellement de mot 2
  function cherche (ligneCode, i) {
    let j
    let continuer = true
    while (i < ligneCode.length && continuer) {
      for (j = 2; j < arguments.length; j++) {
        if (ligneCode[i] == arguments[j]) { continuer = false }
      }
      if (continuer) { i++ }
    }
    return i
  }

  function trouve (ligneCode, i, mot) {
    return cherche(ligneCode, i, mot) < ligneCode.length
  }

  function chercheFonction (s, contenu) {
    let parenth = 0; let parMax = 0; let i; let c = ''; let f = ''; let x = ''
    for (i = 0; i < s.length; i++) {
      c = s.charAt(i)
      switch (c) {
        case '(' :
          parenth++
          parMax = Math.max(parenth, parMax)
          break
        case ')' :
          parenth--
          break
        default :
          switch (parenth) {
            case 0:
              f += c
              break
            case 1 :
              x += c
              break
            default :
              break
          } // parenth
      } // c
    } // for
    if (parenth == 0 && parMax == 1 && f != '' && x != '') {
      setVar(f, 'non définie')
      return f + '=function(' + x + "){this.code='" + contenu + "';return " + contenu + '};'
    }
    setVar(s, 'non définie')
    return s + '=' + contenu + ';'
  }

  function parseAffecter (ligneCode) {
    const mots = { affecter: 1, stocker: 1, mettre: 1 }
    let contenu = ''; let js = ''; let j
    const ifleche = cherche(ligneCode, 1, '→')
    if (ifleche < ligneCode.length - 1) {
      ligneCode[ifleche] = gras(ligneCode[ifleche])
      for (let i = 0; i < ifleche; i++) {
        contenu += traduit(ligneCode[i])
      }
      return chercheFonction(ligneCode[ifleche + 1], contenu)
    }
    if ((ligneCode[0] in mots)) {
      ligneCode[0] = gras(ligneCode[0])
      j = cherche(ligneCode, 3, 'valeur')
      if (ligneCode[1] == 'à' || ligneCode[1] == 'dans') {
        if (j < ligneCode.length - 1) {
          ligneCode[j] = gras(ligneCode[j])
          ligneCode[1] = gras(ligneCode[1])
          return chercheFonction(ligneCode[2], traduit(ligneCode[j + 1]))
        } else { return false }
      } else {
        if (j < ligneCode.length - 1) {
          js = chercheFonction(ligneCode[3], traduit(ligneCode[j + 1]))
        } else {
          if (ligneCode[2] == 'à' || ligneCode[2] == 'dans') { ligneCode[2] = gras(ligneCode[2]) }
          js = chercheFonction(ligneCode[3], traduit(ligneCode[1]))
        }
        return js
      }
    }
    const i = cherche(ligneCode, 1, 'prend')
    if (i < ligneCode.length - 2) {
      j = cherche(ligneCode, i, 'valeur')
      if (j < ligneCode.length - 1) {
        ligneCode[i] = gras(ligneCode[i])
        ligneCode[j] = gras(ligneCode[j])
        return chercheFonction(ligneCode[i - 1], traduit(ligneCode[j + 1]))
      }
    }
    return false
  }

  // tableaux
  function parseAjouter (ligneCode) {
    const mots = { ajouter: 1, soustraire: 2, multiplier: 3, diviser: 4 }
    if ((!(ligneCode[0] in mots) || ligneCode[2] != 'à') && ligneCode[2] != 'dans' && ligneCode[2] != 'par') { return false }
    const mot = ligneCode[0]
    ligneCode[0] = gras(ligneCode[0])
    ligneCode[2] = gras(ligneCode[2])
    if (mots[mot] < 3) { return 'ALajouter(' + traduit(ligneCode[1]) + ',' + traduit(ligneCode[3]) + ',' + mots[mot] + ');' }
    return 'ALajouter(' + traduit(ligneCode[3]) + ',' + traduit(ligneCode[1]) + ',' + mots[mot] + ');'
  }

  function parseAfficher (ligneCode) {
    const mots = { afficher: 1, ecrire: 1, écrire: 1 }
    if (!(ligneCode[0] in mots)) { return false }
    let js = 'ALafficher('
    texteAffiche = ''
    ligneCode[0] = gras(ligneCode[0])
    for (let i = 1; i < ligneCode.length; i++) {
      if (i > 1) { js += ',' }
      js += "'" + traduit(ligneCode[i]) + "'"
      texteAffiche += ligneCode[i] + ' '
    }
    js += ');'
    return js
  };

  function parseDemander (ligneCode) {
    const mots = { demander: 1, saisir: 1, lire: 1, entrer: 1, choisir: 1 }
    if (!(ligneCode[0] in mots)) { return false }
    ligneCode[0] = gras(ligneCode[0])
    let js = ''; let j
    if (texteAffiche === '') {
      for (j = 1; j < ligneCode.length; j++) {
        js += ligneCode[j] + "=ALdemander('Entrer','" + ligneCode[j] + "');"
        setVar(ligneCode[j], 'Non définie')
      }
    } else {
      for (j = 1; j < ligneCode.length; j++) {
        js += ligneCode[j] + "=ALdemander('" + texteAffiche + '\\nEntrer ' + ligneCode[j] + "');"
        setVar(ligneCode[j], 'Non définie')
      }
    }
    return js
  };

  function parseSi (ligneCode) {
    if (ligneCode[0] != 'si') { return false }
    ligneCode[0] = gras(ligneCode[0])
    let js = 'if('
    for (let i = 1; i < ligneCode.length; i++) {
      if (ligneCode[i] != 'alors') { js += traduit(ligneCode[i]) } else { ligneCode[i] = gras(ligneCode[i]) }
    }
    js += '){'
    return js
  };

  // enleve le premier terme si c’est le mot "alors" ou le mot "faire"
  // ces mots n’ont pas d’équivalent javascript
  function parseAlorsFaire (ligneCode) {
    if (ligneCode[0] != 'alors' && ligneCode[0] != 'faire') { return ['', ligneCode] }
    const s = gras(ligneCode[0])
    if (ligneCode.length == 1) { ligneCode[0] = '' } else { ligneCode.shift() }
    return [s, ligneCode]
  };

  function parseSinon (ligneCode) {
    if (ligneCode[0] != 'sinon') { return false }
    ligneCode[0] = gras(ligneCode[0])
    const js = '}else{'
    return js
  };

  function parseFin (ligneCode) {
    if (ligneCode[0] == 'finsi' || ligneCode[0] == 'finpour' || ligneCode[0] == 'fsi' || ligneCode[0] == 'fpour') {
      ligneCode[0] = gras(ligneCode[0])
      return '}'
    }
    if (ligneCode[0] == 'fin' && (trouve(ligneCode, 1, 'si') || trouve(ligneCode, 1, 'pour'))) {
      ligneCode[0] = gras(ligneCode[0])
      let i = cherche(ligneCode, 1, 'si')
      if (i >= ligneCode.length) { i = cherche(ligneCode, 1, 'pour') }
      if (i < ligneCode.length) { ligneCode[i] = gras(ligneCode[i]) }
      return '};'
    }
    return false
  };

  function parseTantQue (ligneCode) {
    const mots = { tant: 1, tantque: 1 }
    if (!(ligneCode[0] in mots)) { return false }
    let d = 1
    if (ligneCode[0] == 'tant') {
      ligneCode[1] = gras(ligneCode[1])
      d++
    }
    ligneCode[0] = gras(ligneCode[0])
    let js = 'while('
    for (let i = d; i < ligneCode.length; i++) {
      if (ligneCode[i] != 'faire') { js += traduit(ligneCode[i]) } else { ligneCode[i] = gras(ligneCode[i]) }
    }
    return js + '){'
  }

  function parseFinTantQue (ligneCode) {
    if ((ligneCode[0] == 'fintantque' || ligneCode[0] == 'ftantque' || ligneCode[0] == 'fin') && (trouve(ligneCode, 1, 'tant') || trouve(ligneCode, 1, 'tantque'))) {
      ligneCode[0] = gras(ligneCode[0])
      let i = cherche(ligneCode, 1, 'tant')
      if (i < ligneCode.length - 1) {
        ligneCode[i] = gras(ligneCode[i])
        ligneCode[i + 1] = gras(ligneCode[i + 1])
      } else {
        i = cherche(ligneCode, 1, 'tantque')
        if (i < ligneCode.length - 1) { ligneCode[i] = gras(ligneCode[i]) }
      }
      return '};'
    }
    return false
  }

  function parsePour (ligneCode) {
    if (ligneCode[0] != 'pour') { return false }
    ligneCode[0] = gras(ligneCode[0])
    let js = 'for (' + ligneCode[1] + '='
    setVar(ligneCode[1], 'Non définie')
    let i = 2
    i = cherche(ligneCode, i, 'de')
    i++
    if (i >= ligneCode.length) { return false }
    ligneCode[i - 1] = gras(ligneCode[i - 1])
    js += ligneCode[i] + ';'
    i = cherche(ligneCode, i, 'jusque', 'jusqu`à', 'à')
    i++
    if (i >= ligneCode.length) { return false }
    ligneCode[i - 1] = gras(ligneCode[i - 1])
    js += ligneCode[1] + '<=' + traduit(ligneCode[i]) + ';' + ligneCode[1] + '++){'
    i = cherche(ligneCode, i, 'faire')
    if (i < ligneCode.length) { ligneCode[i] = gras(ligneCode[i]) }
    return js
  }

  function parseRepeter (ligneCode) {
    if (ligneCode[0] != 'répéter') { return false }
    ligneCode[0] = gras(ligneCode[0])
    return 'do{'
  }

  function parseJusque (ligneCode) {
    let i = cherche(ligneCode, 0, 'jusque', 'jusqu`à')
    if (i != 0) { return false }
    ligneCode[0] = gras(ligneCode[0])
    let js = '} while(!('
    for (i = 1; i < ligneCode.length; i++) { js += traduit(ligneCode[i]) }
    js += '));'
    return js
  }

  function modeTortue (activer) {
    if (((!w.ALtortueActive && activer) || !activer) && w.ALtortueActive) {
      modeTortue()
    }
  }

  function parseAvancer (ligneCode) {
    return parse(ligneCode, { avancer: 1, avance: 1 })
  }
  function parse (ligneCode, mots) {
    if (!(ligneCode[0] in mots)) { return false }
    modeTortue(true)
    ligneCode[0] = gras(ligneCode[0])
    let js = 'ALtortue.avance('
    const i = cherche(ligneCode, 1, 'de')
    if (i >= ligneCode.length) {
      js += ligneCode[1]
    } else {
      js += ligneCode[i + 1]
      ligneCode[i] = gras(ligneCode[i])
    }
    js += ');'
    return js
  }

  function parseReculer (ligneCode) {
    const mots = { reculer: 1, recule: 1 }
    if (!(ligneCode[0] in mots)) { return false }
    modeTortue(true)
    ligneCode[0] = gras(ligneCode[0])
    let js = 'ALtortue.avance('
    const i = cherche(ligneCode, 1, 'de')
    if (i >= ligneCode.length) { js += '-' + ligneCode[1] } else {
      js += '-' + ligneCode[i + 1]
      ligneCode[i] = gras(ligneCode[i])
    }
    js += ');'
    return js
  }

  function parseTourner (ligneCode) {
    const mots = {
      tourner: 1,
      tourne: 1
    }
    if (!(ligneCode[0] in mots)) { return false }
    modeTortue(true)
    ligneCode[0] = gras(ligneCode[0])
    let js = 'ALtortue.tourne('
    let i = cherche(ligneCode, 1, 'droite')
    if (i < ligneCode.length) {
      js += '-' // tourner à droite
    } else {
      i = cherche(ligneCode, 1, 'gauche')
    }
    if (i < ligneCode.length) {
      ligneCode[i] = gras(ligneCode[i])
    } else {
      i = 0
    }
    const j = cherche(ligneCode, 1, 'de')
    if (j < ligneCode.length) {
      ligneCode[i] = gras(ligneCode[i])
      i = Math.max(i, j)
    }
    js += ligneCode[i + 1]
    js += ');'
    return js
  }

  function parseSauter (ligneCode) {
    return parse(ligneCode, { sauter: 1, saute: 1 })
  }

  function parseManger (ligneCode) {
    if (!(ligneCode[0] == 'manger' || ligneCode[0] == 'mange' || ligneCode[0] == 'départ')) { return false }
    modeTortue(true)
    ligneCode[0] = gras(ligneCode[0])
    return 'ALtortue.mange();'
  }

  function parseDessiner (ligneCode) {
    const mots = { point: 1, segment: 1, cercle: 1, rectangle: 1, couleur: 1 }
    if (!(ligneCode[0] in mots)) { return false }
    let js = ''
    dessinActif = true
    switch (ligneCode[0]) {
      case 'point' :
        js = 'ALrep.point(' + ligneCode[1] + ',' + ligneCode[2] + ');'
        break
      case 'segment' :
        js = 'ALrep.segment(' + ligneCode[1] + ',' + ligneCode[2] + ',' + ligneCode[3] + ',' + ligneCode[4] + ');'
        break
      case 'rectangle' :
        js = 'ALrep.rect(' + ligneCode[1] + ',' + ligneCode[2] + ',' + ligneCode[3] + ',' + ligneCode[4] + ',false);'
        break
      case 'cercle' :
        js = 'ALrep.cercle(' + ligneCode[1] + ',' + ligneCode[2] + ',' + ligneCode[3] + ');'
        break
      case 'couleur' :
        js = "ALrep.stroke('" + ligneCode[1] + "');"
        break
    }
    ligneCode[0] = gras(ligneCode[0])
    return js
  }

  let faireFichier = true
  function parseService (ligneCode) {
    let js
    switch (ligneCode[0]) {
      case 'démo' :
        js = 'ALservice(0);'
        break
      case 'repas' :
        js = 'ALservice(1);'
        faireFichier = false
        break
      case 'itinéraire' :
        js = 'ALservice(2);'
        faireFichier = false
        break
      case 'démo_repas' :
        js = 'ALservice(3);'
        break
      case 'démo_itinéraire' :
        js = 'ALservice(4);'
        break
      default :
        return false
    }
    ligneCode[0] = gras(ligneCode[0])
    return js
  }

  function parseControle (ligneCode) {
    if (ligneCode[0] != 'contrôler') { return false }
    controles = ligneCode[1]
  }

  function parseDebutFin (ligneCode) {
    if ((ligneCode.length > 1 || ligneCode[0] != 'début_algo') && ligneCode[0] != 'fin_algo') {
      return false
    }
    return 'ALrien();'
  }

  // laisser parseDemander au début
  const fonctions = [parseDemander, parseAffecter, parseAjouter, parseAfficher, parseSi, parseSinon, parseFin, parseTantQue, parseFinTantQue, parsePour, parseRepeter, parseJusque, parseAvancer, parseReculer, parseTourner, parseSauter, parseManger, parseDessiner, parseService, parseControle, parseDebutFin]
  const indentations = [[0, 0], [0, 0], [0, 0], [0, 0], [0, 2], [-2, 2], [-2, 0], [0, 2], [-2, 0], [0, 2], [0, 2], [-2, 0], [0, 0], [0, 0], [0, 0], [0, 0], [0, 0], [0, 0], [0, 0], [0, 0], [0, 0]]

  function normalise (s) {
    // enlève les espaces multiples et  les espaces dans les parenthèses
    let t = ''; let parenthese = 0; let dejaBlanc = false; let c
    for (let i = 0; i < s.length; i++) {
      c = s.charAt(i)
      if (c == '(') { parenthese++ }
      if (c == ')') { parenthese-- }
      if (parenthese > 0) { dejaBlanc = true }
      if (c == ' ') {
        if (!dejaBlanc) {
          dejaBlanc = true
          t += c
        }
      } else {
        dejaBlanc = false
        t += c
      }
    }
    // a éviter si ok
    s = t
    s = s.toLowerCase()
    s = s.replace(/&lt;/gi, '<')
    s = s.replace(/&gt;/gi, '>')
    s = s.replace(/&le;/gi, '<=')
    s = s.replace(/&ge;/gi, '>=')
    s = s.replace(/'/g, '`')
    return s
  }

  // coloration des variables
  function coloreVar (s) {
    function spe (c) {
      const t = ['+', '-', '/', '*', '%', '(', ')', '[', ']', '<', '>', '=', '≤', '≥', '≠', '→', 'π']
      for (let i = 0; i < t.length; i++) {
        if (c == t[i]) { return true }
      }
      return false
    }
    let j = 0; let s1 = ''; let c
    while (j < s.length) {
      c = ''
      while (j < s.length && !spe(s.charAt(j))) {
        c += s.charAt(j)
        j++
      }
      if (c in vars) { c = '££v' + c + '££w' }
      s1 += c
      while (j < s.length && spe(s.charAt(j))) {
        s1 += s.charAt(j)
        j++
      }
    }
    return s1
  }

  modeTortue(false)
  vars = {}
  numLignes = []
  dessinActif = false

  // charger le texte de l’algorithme
  let prog = litSource()
  prog = normalise(prog)
  const codes = []
  let lignes = prog.split('//br;')

  // initialisation du pas à pas
  if (!preservePasApas) {
    // non pas à pas
    papReset()
    document.getElementById('ALpasapas').style.display = 'none'
    // enleve début_algo et fin_algo
    if (lignes[0] == 'début_algo') { lignes.shift() }
    if (lignes[lignes.length - 1] == 'fin_algo') { lignes.pop() }
  } else {
    //  pas à pas
    // introduit les lignes "début_algo" et "fin algo"
    if (lignes[0] != 'début_algo') {
      if (lignes[0] == '' || lignes[0] == ' ') { lignes[0] = 'début_algo' } else {
        const debut = ['début_algo']
        lignes = debut.concat(lignes)
      }
    }
    if (lignes[lignes.length - 1] == '' || lignes[lignes.length - 1] == ' ') { lignes[lignes.length - 1] = 'fin_algo' } else if (lignes[lignes.length - 1] != 'fin_algo') { lignes.push('fin_algo') }
  }
  film = []
  filmTortueEfface()
  filmTortueSetElt([0])
  filmDessinEfface()
  filmDessinSetElt([0])

  // suppression des lignes vides
  for (let i = lignes.length - 1; i >= 0; i--) {
    if (lignes[i] == '') {
      lignes.splice(i, 1)
    }
  }

  for (let i = 0; i < lignes.length; i++) {
    // elimine blancs au début et à la fin
    lignes[i] = lignes[i].replace(/^\s+/g, '').replace(/\s+$/g, '')
    codes.push(lignes[i].split(' '))
  }
  // traduction algo en javascript, coloration et indentation du source
  let j, k
  const tabjs = []
  let indente = 0
  let indentation = ''
  let tab
  let retour = []
  w.ALformatees = []
  let ligne = ''

  for (let i = 0; i < codes.length; i++) {
    retour = false
    tab = parseAlorsFaire(codes[i]) // elimine "alors" et "faire"
    for (let j = 0; j < fonctions.length; j++) {
      retour = fonctions[j](tab[1])
      // gestion du prompt sur 2 lignes
      if (j == 0) {
        texteAffiche = ''
      }
      if (retour) {
        if (preservePasApas) {
          tabjs.push(retour + ' ALfilmer(' + i + ');')
        } else {
          tabjs.push(retour + 'ALcompte();')
        }
        numLignes.push(i)
        break
      }
    }
    if (tab[0] != '') {
      // on réintègre "alors" ou "faire"
      codes[i].unshift(tab[0])
    }

    if (!retour || retour == 'ALrien();') {
      ligne = codes[i].join(' ')
      indentation = ''
      // correction de l’indentation
      // si la ligne commence par "alors" ou "faire"
      let indenteAlors
      if (tab[0] != '') {
        indenteAlors = Math.max(0, indente - 2)
      } else {
        indenteAlors = indente
      }
      for (k = 0; k < indenteAlors; k++) indentation += ' '
      if (!retour) {
        w.ALformatees.push(indentation + "<span class='comm'>" + special(ligne) + '</span>')
      } else {
        w.ALformatees.push(indentation + "<span class='deb'>" + special(ligne) + '</span>')
      }
    } else {
      indente += indentations[j][0]
      indentation = ''
      for (k = 0; k < indente; k++) { indentation += ' ' }
      for (k = 0; k < codes[i].length; k++) { codes[i][k] = coloreVar(codes[i][k]) }
      w.ALformatees.push(indentation + special(codes[i].join(' ')))
      indente += indentations[j][1]
    }
  }
  if (w.ALtortueActive) {
    ALnouvelleTortue()
  }
  ecritSource(w.ALformatees.join('<br>'))
  if (faireFichier) { faitFichier() } // préparation sauvegarde
  return tabjs.join('\n')
}

function pasApas () {
  if (clic) return
  // empêcher les ralentis tortue simultanés
  if (w.ALtortueActive) {
    const d = new Date()
    if (indexPaP) {
      if (d - indexPaP < getFreqTortue() * 40) { return }
    }
    indexPaP = d
  }
  voirDiv('ALaide', false)
  papInc()
  const indexPap = papGetIndex()
  if (indexPap == 0) {
    try {
      lancer(true)
      if (!w.ALtimer) {
        // eslint-disable-next-line no-alert
        alert('Vous pourvez maintenant voir le déroulement de l’algorithme.\nen appuyant à nouveau sur la trace de pas autant de fois qu’il le faudra.')
      }
      if (w.ALtortueActive) { ALnouvelleTortue() }
      if (dessinActif) w.ALrep = new ALrepere()
    } catch (err) {
      console.error(err)
    }
  }
  if (indexPap >= film.length) {
    if (w.ALtimer) {
      const enBoucle = document.getElementById('ALenBoucle')
      if (!enBoucle.checked) { clearInterval(w.ALtimer) }
    }
    parseAlgo()
  }
  let i
  if (indexPap > -1) {
    i = film[indexPap][0]
  } else {
    return
  }
  const pasapas = document.getElementById('ALpasapas')
  pasapas.style.display = 'block'
  pasapas.innerHTML = '<b>Variables et fonctions</b><br>' + film[indexPap][1]
  //  surligne l’instruction
  souligne(i, 'pas')
  if (w.ALtortueActive) {
    const filmElt = filmTortueGetElt(indexPap)
    switch (filmElt[0]) {
      case 0:
        break
      case 1:
        w.ALtortue.tourneRalenti(filmElt[1])
        break
      case 2:
        w.ALtortue.avanceRalenti(filmElt[1], filmElt[2])
        break
      case 3:
        w.ALtortue.mange()
        break
    }
  }
  // dessin
  if (dessinActif) {
    const filmDessinElt = filmDessinGetElt(indexPap)
    switch (filmDessinElt[0]) {
      case 0:
        break
      case 1:
        w.ALrep.stroke(filmDessinElt[1])
        break
      case 2:
        w.ALrep.segment(filmDessinElt[1], filmDessinElt[2], filmDessinElt[3], filmDessinElt[4])
        break
      case 3:
        w.ALrep.cercle(filmDessinElt[1], filmDessinElt[2], filmDessinElt[3])
        break
      case 4:
        w.ALrep.rect(filmDessinElt[1], filmDessinElt[2], filmDessinElt[3], filmDessinElt[4])
        break
      case 5:
        w.ALrep.point(filmDessinElt[1], filmDessinElt[2])
        break
    }
  }
}

function pasApasAuto (elt) {
  function progress (delai) {
    const pbar = document.getElementById('ALpbar')
    let pcnt = 0
    const intervalId = setInterval(function () {
      pcnt += 2
      if (pcnt <= pbar.max) {
        pbar.value = pcnt
      } else {
        pbar.value = 0
        clearInterval(intervalId)
      }
    }, delai / 50)
  }
  let delai = document.getElementById('ALdelai').value
  delai = parseInt(delai) * 1000
  progress(delai)
  setTimeout(metronome, delai)
}

function setVar (nom, valeur) {
  if (nom.indexOf('[') > -1) { return }
  vars[nom] = valeur
  w.ALvarAnciennes[nom] = valeur
  w[nom] = valeur
}

function souligne (num, nomClasse) {
  const ligne = w.ALformatees[num]
  w.ALformatees[num] = "<span id='aScroller' class='" + nomClasse + "'>" + ligne + '</span>'
  ecritSource(w.ALformatees.join('<br>'))
  w.ALformatees[num] = ligne
  if (document.getElementById('aScroller')) {
    document.getElementById('aScroller').scrollIntoView(false)
  }
}

function special (prog) {
  const t = ['<', '>', '££b', '££c', '££v', '££w']
  const u = ['&lt;', '&gt;', "<span class='mot'>", '</span>', "<span class='variable'>", '</span>']
  let reg
  for (let i = 0; i < t.length; i++) {
    reg = RegExp(t[i], 'g')
    prog = prog.replace(reg, u[i])
  }
  return prog
}

function stopMetronome () {
  setFreqTortue(25)
  if (!w.ALtimer) { return false }
  clearInterval(w.ALtimer)
  const enBoucle = document.getElementById('ALenBoucle')
  if (!enBoucle.checked) { enBoucle.checked = false }
  parseAlgo()
  if (w.ALtortue) w.ALtortue.reinitialise()
  return true
}

function voirDiv (div, ouvrir) {
  if (div.charAt) { div = document.getElementById(div) }
  if ((div.style.display === 'block' || arguments.length === 2) && ouvrir == false) {
    div.style.display = 'none'
  } else {
    div.style.display = 'block'
  }
}

// mise à jour de vars (jamais utilisé)
// function ALvinitialiseVars () {
//   for (const prop in Object.key(vars)) {
//     vars[prop] = null
//   }
// }

w.ALafficher = afficher
w.ALchargeDemo = function chargeDemo (sel) {
  const d = document.getElementById('ALgauche')
  const elt = chercheElement('ALdemos', 'PRE', sel.options.selectedIndex)
  d.innerHTML = '<pre>\n' + elt.innerHTML + '\n</pre>'
  parseAlgo()
}
w.ALcompte = compte
w.ALfaitPrecision = faitPrecision
w.ALfilmer = function filmer (numLigne) {
  if (papGetIndex() === -2) {
    // on filme seulement en mode pas à pas
    return
  }
  compte()
  const ligne = []
  let message = ''
  let nom, valeur, ajout
  // numero de la ligne du programme
  ligne.push(numLigne)
  // vars est la liste des noms des variables utilisateur
  for (nom in vars) {
    try {
      // w[nom] est la valeur de la variable dont le nom est donné
      valeur = w[nom]
      vars[nom] = valeur
    } catch (err) {
      valeur = 'non définie'
    }
    if ((typeof valeur) === 'function') {
      // eslint-disable-next-line new-cap
      const g = new valeur()
      valeur = g.code
    }
    if (String(vars[nom]) !== String(w.ALvarAnciennes[nom])) {
      ajout = "<span class='barre'>" + nom + ' : ' + w.ALvarAnciennes[nom] + '</span><br>'
      ajout += "<span class='change'>" + nom + ' : ' + valeur + '</span>'
      w.ALvarAnciennes[nom] = vars[nom]
    } else { ajout = nom + ' : ' + valeur }
    ajout += '<br>'
    message += ajout
  }
  ligne.push(message)
  film.push(ligne)
  filmTortueSetElt([0])
  filmDessinSetElt([0])
}
w.ALparseAlgo = parseAlgo
// correction du bug qui supprime le pre
w.ALpre = function pre (div, evt) {
  if (evt.keyCode != 8) { // backspace
    return
  }
  const elt = div.lastChild
  if (!elt || !elt.tagName) {
    ALinsere('<pre>&nbsp;</pre>')
  }
}

// function ALtransfere (elt) {
//   let texte = elt.textContent
//   texte = special(texte)
//   if (confirm('Transférer ce programme à droite en écrasant le programme courant ?')) {
//     ecritSource(texte)
//     ALformate()
//   }
// }

w.ALajouter = function ALajouter (y, x, op) {
  if (x.push && op == 1) { x.push(y) } else {
    switch (op) {
      case 1:
        w[x] += y
        break
      case 2:
        w[x] -= y
        break
      case 3:
        w[x] *= y
        break
      case 4:
        w[x] /= y
        break
    }
  }
}

// function ALretour () {
//   document.getElementById('doc').style.display = 'none'
//   document.getElementById('aide').style.display = 'block'
//   document.getElementById('ALRetour').style.display = 'none'
// }

// function ALvaEtVient (elt, fermer) {
//   let e = elt.nextSibling
//   while (!e.tagName || e.tagName.toUpperCase() != 'DIV') { e = e.nextSibling }
//   if (e.style.display == 'none' && !fermer) {
//     e.style.display = 'block'
//     elt.scrollIntoView(true)
//   } else {
//     e.style.display = 'none'; b
//     e = e.firstChild; outons
//     while (e) {
//       ALvaEtVient(e, true)
//       e = e.nextSibling
//     }
//   }
// }

w.ALmail = function ALmail () {
  let chaine_mail = 'mailto:?subject= Programme javascript'
  chaine_mail += '&body='
  let s = litSource(true)
  const t1 = ['<br>', '&lt;', '&gt;', '&le;', '&ge;', '&ne;', '&pi;', '&rarr;', '&']
  const t2 = ['\n', '<', '>', '≤', '≥', '≠', 'π', '→', '\\&']
  for (const [i, t] of t1.entries()) {
    const reg = new RegExp(t, 'gi')
    s = s.replace(reg, t2[i])
  }
  chaine_mail += encodeURIComponent(s)
  location.href = chaine_mail
}

w.ALservice = function ALservice (i) {
  let s
  switch (i) {
    case 0: // construction d’une démo
      s = litSource(true)
      var reg
      var t1 = ['<', '>', '≤', '≥', '≠', 'π', '→', 'démo', 'taper un algorithme ici :\n', '\n']
      var t2 = ['&lt;', '&gt;', '&le;', '&ge;', '&ne;', '&pi;', '&rarr;', '', '', '\\n']
      for (let i = 0; i < t1.length; i++) {
        reg = new RegExp(t1[i], 'gi')
        s = s.replace(reg, t2[i])
      }
      s = '"' + s + '"'
      break
    case 1: // construction d’un repas tortue à sauvgarder
      faitFichier('repas', '[' + w.ALtortue.chaineRepas + ']')
      break
    case 2: // construction d’un itinéraire pour la tortue à sauvgarder
      faitFichier('itineraire', w.ALtortue.chaineDessin + ']')
      break
    case 3: // construction d’un repas tortue à intégrer dans le source
      s = 'Dans le code-source de PSyLVIA, cherchez le mot lesRepas,\npuis collez à l’endroit indiqué l’instruction suivante :\n\nlesRepas.push([' + w.ALtortue.chaineRepas + ']);'
      break
    case 4: // construction d’un itinéraire pour la tortue à intégrer dans le source
      s = 'Dans le code-source de PSyLVIA, cherchez le mot lesDessins,\npuis collez à l’endroit indiqué l’instruction suivante :\n\nlesDessins.push(' + w.ALtortue.chaineDessin + ']);'
      break
  }
  if (i == 0 || i > 2) {
    const w2 = w.open()
    const newDoc = w2.document.open('text')
    newDoc.write(s)
    newDoc.close()
  }
}

w.ALimprime = function ALimprime () {
  const w2 = w.open()
  const s = document.getElementById('ALgauche').innerHTML
  w2.document.write(s)
  w2.print()
}

// sauvegarde d’urgence par cookie
// function ALecrireCookie (nom, val) {
//   const date = new Date()
//   date.setHours(date.getHours() + 1)
//   // date.setFullYear(date.getFullYear()+1);
//   document.cookie = nom + '=' + escape(val) + ';expires=' + date
// }

// function ALlireCookie (nom) {
//   let s = document.cookie
//   let i = s.indexOf(nom, 0)
//   if (i == -1) { return '' }
//   s = s.substring(i)
//   i = s.indexOf('=', 0)
//   const j = s.indexOf(';', 0)
//   if (j != -1) { s = s.substring(i + 1, j) } else { s = s.substring(i + 1) }
//   return unescape(s)
// }

// pas utilisé dans le contexte j3p
// w.ALsauveUrgence = function ALsauveUrgence () {
//   let donnee = litSource()
//   donnee = special(donnee)
//   const t = donnee.split('//br;')
//   if (t[0] == 'début_algo') { t.shift() }
//   if (t[t.length - 1] == 'fin_algo') { t.pop() }
//   donnee = t.join('//br;')
//   // sauvegarde d’urgence par cookie valable 1 heure
//   ALecrireCookie('psylvia_urgence', donnee)
// }

// pas utilisé dans le contexte j3p
// function ALrestaureUrgence () {
//   const s = ALlireCookie('psylvia_urgence')
//   if (!s) { return false }
//   if (confirm("Sauvegarde d’urgence de l’algorithme détectée.\nRestaurer ?")) {
//     ecritSource(s)
//     parseAlgo()
//     return true
//   }
//   return false
// }

// sauvegarde par lecture/écriture sur le disque

w.ALouvrir = function ALouvrir () {
  document.getElementById('ALstock').style.display = 'block'
}

w.ALsauveIE = function sauveIE () {
  const ifr = document.getElementById('ALiframe')
  ifr.contentWindow.document.r = ifr.contentWindow.document.execCommand('SaveAs', false, 'algo.txt')
  if (!ifr.contentWindow.document.r) {
    // eslint-disable-next-line no-alert
    alert('Echec de la sauvegarde')
  }
}

w.ALlitFichier = function litFichier (fileInput) {
  try {
    const reader = new FileReader()
    reader.readAsText(fileInput.files[0])
    reader.onload = function () {
      let s = reader.result
      s = s.replace(/\n/g, ' ')
      s = s.replace(/\r/g, '')
      const json = JSON.parse(s)
      const initTortue = [300, 300]
      if (json.algo) {
        verifs = {}
        if (json.controles) {
          verifs = json.controles
        }
        ecritSource(json.algo)
        parseAlgo()
      } else
        if (json.repas) {
          modeTortue()
          Tortue.prototype.repasLu = true
          // eslint-disable-next-line no-eval
          Tortue.prototype.repas = eval(json.repas)
          // bug ???---------------------------------------
          Tortue.prototype.dessin = initTortue
          w.ALtortue.positionneImg()
        }
      if (json.itineraire) {
        modeTortue()
        Tortue.prototype.dessinLu = true
        // eslint-disable-next-line no-eval
        Tortue.prototype.dessin = eval(json.itineraire)
        // bug ??? ---------------------------------------
        Tortue.prototype.repas = initTortue
        w.ALtortue.dessineItineraire()
        w.ALtortue.positionneImg()
      }
    }
  } catch (e) {
    console.error(e)
    j3pShowError('fonction non supportée par le navigateur : ' + e.message)
  }
  document.getElementById('ALstock').style.display = 'none'
}

// fin sauvegarde

w.ALfermer = function ALfermer (id) {
  const n = document.getElementById(id)
  n.style.display = 'none'
}

w.ALeffacer = function ALeffacer (sortie) {
  // let message
  // if (sortie) {
  //   message = 'Effacer les sorties'
  // } else {
  //   message = 'Effacer l’algorithme'
  // }
  // if (!confirm(message)) return
  if (sortie) {
    const b = document.getElementById('ALdroite')
    b.innerHTML = '<p><b>Sorties</b></p>'
    if (w.ALtortueActive) { ALnouvelleTortue() }
    return
  }
  const h = document.getElementById('ALgauche')
  h.innerHTML = ''
  h.focus()
  ALinsere('<pre>Taper un algorithme ici:<br></pre>')
  parseAlgo()
  h.blur()
  document.getElementById('ALpasapas').style.display = 'none'
  verifs = {}
}

// convertit une chaine en nombre SI PERTINENT
function ALnombre (s) {
  const n = parseInt(s)
  const x = parseFloat(s)
  if (isNaN(n) && isNaN(x)) { return s }
  if (x == n) { return n }
  return x
}

w.ALdemander = function ALdemander () {
  let message = ''
  for (let i = 0; i < arguments.length; i++) { message += arguments[i] + ' ' }
  // eslint-disable-next-line no-alert
  const x = prompt(message)
  if (!x) {
    w.ALstop = true
    // introduire une erreur avec throw à la place
    throw Error('Programmme arrêté par l’utilisateur')
    // eval("x=avorte programme");
  }
  const s = x + ' '
  const t = s.split(/[,;]+/g)
  if (t.length < 2) { return ALnombre(x) }
  for (let i = 0; i < t.length; i++) {
    t[i] = ALnombre(t[i])
  }
  return t
}

// function ALelement (type, cible, classe) {
//   const elem = document.createElement(type)
//   if (classe) { elem.className = classe }
//   cible.appendChild(elem)
//   return elem
// }

function ALinsere (html) {
  const ie = w.ActiveXObject
  if (!ie) {
    document.execCommand('insertHTML', false, html)
    document.getElementById('ALgauche').focus()
  } else {
    // IE9 and non-IE
    const sel = w.getSelection()
    if (sel.getRangeAt && sel.rangeCount) {
      let range = sel.getRangeAt(0)
      range.deleteContents()
      // Range.createContextualFragment() would be useful here but is
      // non-standard and not supported in all browsers (IE9, for one)
      const el = document.createElement('div')
      el.innerHTML = html
      const frag = document.createDocumentFragment(); let node; let lastNode
      while ((node = el.firstChild)) {
        lastNode = frag.appendChild(node)
      }
      range.insertNode(frag)
      // Preserve the selection
      if (lastNode) {
        range = range.cloneRange()
        range.setStartAfter(lastNode)
        range.collapse(true)
        sel.removeAllRanges()
        sel.addRange(range)
      }
    }
  }
}

w.ALsup = function ALsup (b) {
  document.execCommand('superscript', false, null)
  document.getElementById('ALgauche').focus()
}

function ALafficheBalai (cible) {
  const img = document.getElementById(cible)
  img.setAttribute('src', '')
}

// fonctions mathématiques préprogrammées

w.abs = function abs (x) {
  return Math.max(-x, x)
}

w.ent = function ent (x) {
  return Math.floor(x)
}

w.sin = function sin (x) {
  return Math.sin(x)
}

w.cos = function cos (x) {
  return Math.cos(x)
}

w.tan = function tan (x) {
  return Math.tan(x)
}

w.ln = function ln (x) {
  return Math.log(x)
}

w.log = function log (x) {
  return Math.log(x) / Math.log(10)
}

w.exp = function exp (x) {
  return Math.exp(x)
}

w.bernoulli = function bernoulli (p) {
  if (Math.random() < p) { return 1 }
  return 0
}

w.aleatoire = function aleatoire (debut, fin) {
  const ampl = parseFloat(fin) - parseFloat(debut)
  const x = Math.random() * ampl + parseFloat(debut)
  return Math.round(x)
}

w.aleauniforme = function aleauniforme (debut, fin) {
  const ampl = parseFloat(fin) - parseFloat(debut)
  const x = Math.random() * ampl + parseFloat(debut)
  return x
}

w.aleanormal = function aleanormal (m, s) {
  return s * Math.sqrt(-2 * Math.log(Math.random())) * Math.cos(2 * Math.PI * Math.random()) + m
}

// tortue

function ALnouvelleTortue () {
  const droite = document.getElementById('ALdroiteTortue')
  while (droite.firstChild) { droite.removeChild(droite.firstChild) }
  w.ALtortueActive = false
  w.ALtortue = null
  modeTortue()
}

// --------------------- fonctions de dessin --------------------------------

w.ALdimGraph = function ALdimGraph () {
  w.ALfenetre.xMin = document.getElementById('ALxMin').value
  w.ALfenetre.yMin = document.getElementById('ALyMin').value
  w.ALfenetre.xMax = document.getElementById('ALxMax').value
  w.ALfenetre.yMax = document.getElementById('ALyMax').value
  w.ALfenetre.stroke = document.getElementById('ALstroke').value
  w.ALfenetre.axes = document.getElementById('ALaxes').checked
  document.getElementById('ALdimFenetre').style.display = 'none'
}

// initialisation générale

// fonction vide ajoutée pour J3P (qui sera éventuellement initialisée)
w.ALactionFusee = () => undefined

export default function Psylvia () {
  this.aMasquer = {}
  this.demos = {}
  this.precis = 0
  this.haut = 400
}

// réglage de la précision
Psylvia.prototype.precision = function (n) {
  this.precis = n
}

// variable à surveiller
Psylvia.prototype.surveiller = function (nom, valeur) {
  verifs[nom] = valeur
}

// vérifier qu’une variable a bien pris la valeur attendue
Psylvia.prototype.verifier = function (nom) {
  return isVerifOk[nom]
}

// ajouter une démo dans la liste
Psylvia.prototype.ajouteDemo = function (titre, algo) {
  this.demos[titre] = algo
}

// masquer certains élements
Psylvia.prototype.masquer = function () {
  const noms = {
    demos: 'ALselectDemo',
    repere: 'ALparamGraph',
    precision: 'ALprecis',
    ouvrir: 'ALiconeOuvrir',
    enregistrer: 'ALiconeEnregistrer',
    courrier: 'ALiconeCourriel',
    imprimer: 'ALiconeImprimer',
    repas: 'ALrepasTortue',
    balai: 'ALeffaceG',
    tortue: 'ALboutonTortue'
  }
  for (let i = 0; i < arguments.length; i++) { this.aMasquer[noms[arguments[i]]] = true }
}

// action en cas de clic sur la fusee
Psylvia.prototype.actionFusee = function (fonction) {
  actionFusee = fonction
}

Psylvia.prototype.hauteur = function (n) {
  this.haut = Math.max(n, 350)
}

// installation de PSyLVIA dans la page
Psylvia.prototype.installer = function (divGauche, divDroite, algo) {
  if (divGauche.charAt) { divGauche = document.getElementById(divGauche) }
  if (divDroite.charAt) { divDroite = document.getElementById(divDroite) }
  // hauteur des cadres
  divGauche.style.height = this.haut + 'px'
  divDroite.style.height = this.haut + 'px'

  // code HTML récupéré
  // elements droits
  let strVar = ''

  strVar += '<div id="ALdroite">'
  strVar += '      <p><b>Sorties :</b> </p>'
  strVar += '</div>  '

  strVar += '<div id="ALprecis">'
  strVar += 'Précision :'
  strVar += '<input style="width:2em;" id="ALinputPrecis" value="6" onchange="ALfaitPrecision()" > '
  strVar += '</div>'

  strVar += '<div id="ALafficheCoord">'
  strVar += '</div>'

  w.ALdroiteTortueOnClick = function () {
    if (papGetIndex() > -1) voirDiv('ALpasapas')
  }
  strVar += '<div id="ALdroiteTortue" onclick="ALdroiteTortueOnClick()">'
  strVar += '</div>'
  strVar += '<div id="ALrepasTortue">'
  strVar += ' Repas  et itinéraires :'
  w.ALinitialiseTortue = initialiseTortue
  strVar += ' <select id="ALlisteRepas" onchange="ALinitialiseTortue()">'
  strVar += ' </select>'
  strVar += '</div>'

  strVar += '<div id="ALpasapas">'
  strVar += '</div>   '

  strVar += '<img alt="effacer" id="ALeffaceD" title="Effacer les sorties" onclick="ALeffacer(true)"  />'
  strVar += '<!--    AIDE EN LIGNE   -->'
  strVar += '<div id="ALaide">'
  strVar += '<object   data="aide.html"></object>'
  strVar += '</div>'
  strVar += '<div id="ALicones">'
  strVar += '<img id="ALiconeOuvrir" title="Ouvrir un programme" onclick="ALouvrir()" src="" height="20" width="20">'
  strVar += '\n'
  strVar += '<a id="ALiconeEnregistrer"  href="#" id="ALlien"    onmouseover="ALparseAlgo()" download="algorithme.txt">\n'
  strVar += '<img title="Enregistrer le programme"  src="" height="20" width="20">'
  strVar += '</a>'
  strVar += '\n'
  strVar += '<img id="ALiconeCourriel" title="Poster le programme" onclick="ALmail()" src="" height="25" width="25">'
  strVar += '\n'
  strVar += '<img id="ALiconeImprimer" title="imprimer le programme" onclick="ALimprime()" src="" height="20" width="20">'
  strVar += '\n'
  strVar += "<img alt=\"aide\" title=\"Aide\" onclick=\"voirDiv('ALaide'),voirDiv('ALenHaut');\" src=\"\" />"
  strVar += '</div>'
  strVar += '\n'
  strVar += '<div id="ALrythme">'
  strVar += '<p>Pas à pas automatique</p>'
  strVar += 'Une étape par'
  strVar += '<input style="width:2em;" id="ALfreq"  value="1"> seconde(s)<br>'
  strVar += 'Délai intitial'
  strVar += '<input style="width:2em;" value="2" id="ALdelai" > seconde(s)<br>'
  strVar += '<input type="checkbox" id="ALenBoucle" /> En boucle<br>'
  strVar += '<progress id="ALpbar" value="0" min="0" max="100"></progress> '
  w.ALonClickLancer = function () {
    clic = false
    pasApasAuto(this)
  }
  strVar += '<input type="button"  onclick="ALonClickLancer()" value="Lancer" />'
  w.ALonClickFermer = function () {
    clic = false
    this.parentNode.style.display = 'none'
  }
  strVar += '<input type="button" onclick="ALonClickFermer()" value="Fermer" />'
  strVar += '</div>'

  strVar += '<div id="ALstock">'
  strVar += "  <a style=\"position:absolute; top: 2px; right: 10px;\" href=\"javascript:ALfermer('ALstock');\" alt=\"Fermer\">"
  strVar += '  <img title="Fermer" alt="Fermer" src="%3D" height="16" width="16">'
  strVar += '  </a>  '
  strVar += '  <h3>Ouvrir un programme</h3>'
  strVar += '  <p id="ALlisteStock">'
  strVar += '  </p>'
  strVar += '  <input type="file" onchange="ALlitFichier(this)">'
  strVar += '</div>'

  strVar += '<div id="ALdemos" style="display:none">'

  strVar += '<!-- ne pas toucher à ces lignes -->'
  strVar += '<h1>'
  strVar += 'Exemples'
  strVar += '</h1>'
  strVar += '<pre>'
  strVar += 'Taper l’algorithme ici :'
  strVar += '</pre>  '
  strVar += '</div>'

  strVar += '<!-- iframe caché pour sauvegarde sous IE -->'
  strVar += '<iframe id="ALiframe"></iframe>'
  divDroite.innerHTML = strVar

  // -------- remplissage de la div gauche ----------
  strVar = ''
  w.ALonMouseUpGauche = function () {
    // bogue sous firefox, fonction desactivee
    if (isFirefox) { return }
    const b = document.getElementById('ALbex')
    if (document.queryCommandState('superscript')) { b.className = 'exp' } else {
      b.className = ''
    }
  }
  strVar += '<div id="ALgauche" contenteditable="true" spellcheck="false" onmouseup="ALonMouseUpGauche()"  onkeyup="ALpre(this, event)">'
  strVar += '</div>'

  strVar += '<div id="ALboutons">  '

  w.ALonClickExec = () => {
    lancer()
    actionFusee()
  }
  strVar += '<img alt="exe"  onclick="ALonClickExec()" title="Exécuter" src="" /> '

  w.ALonClickPasApas = () => {
    w.AL1clic = false
    if (papGetIndex() === -1) {
      setTimeout(function () {
        if (!stopMetronome()) pasApas()
        w.AL1clic = true
      }, 1000)
    } else {
      pasApas()
    }
  }
  w.ALonDblClickPasApas = () => {
    if (!w.AL1clic && papGetIndex() === -1) {
      clic = true
      voirDiv('ALrythme')
    }
  }
  strVar += '<img alt="Pas a pas" onclick="ALonClickPasApas" ondblclick="ALonDblClickPasApas" title="Exécuter pas à pas" src="" />'

  strVar += "<img id=\"ALparamGraph\" alt=\"graphique\" title=\"Fenêtrage des sorties graphiques hors tortue\" onclick=\"voirDiv('ALdimFenetre');\" src=\"\" />"

  strVar += '<img alt="Mise en forme" title="Colorier et indenter" onclick="ALparseAlgo()"  src="" />'

  w.ALmodeTortue = modeTortue
  strVar += '<img id="ALboutonTortue" alt="tortue" title="Mode tortue" onclick="ALmodeTortue();"  src="" />'
  strVar += '<img alt="effacer" id="ALeffaceG" title="Effacer l’algorithme" onclick="ALeffacer(false)" />'
  strVar += '</div>'
  strVar += '<div id="ALlisteDemo" >'
  strVar += '<select id="ALselectDemo" onchange="ALchargeDemo(this)">'
  strVar += '</select>'
  strVar += '</div>'
  strVar += '<div id="ALtouches">'
  strVar += '<button onclick="ALsup(this);" id="ALbex" title="activer ou désactiver les exposants">'
  strVar += 'x<sup>n</sup>'
  strVar += '</button>\n'
  strVar += "<button  onclick=\"ALinsere('&pi;')\" title=\"insère &pi;\">&pi; </button>\n"
  strVar += "<button  onclick=\"ALinsere('&le;')\" title=\"insère &le;\">&le; </button>\n"
  strVar += "<button  onclick=\"ALinsere('&ge;')\" title=\"insère &ge;\">&ge; </button>\n"
  strVar += "<button  onclick=\"ALinsere('&ne;')\" title=\"insère &ne;\">&ne; </button>\n"
  strVar += "<button  onclick=\"ALinsere('&rarr;')\" title=\"insère &rarr;\">&rarr; </button>"
  strVar += '</div>'

  strVar += '<div id="ALavertissement">'
  strVar += '<h2>Attention !</h2>'
  strVar += '<p>'
  strVar += 'Si vous voyez ce message, cela veut dire que l’exécution des programmes'
  strVar += 'est désactivée sur votre navigateur. '
  strVar += '</p>'
  strVar += '<ul>'
  strVar += '<li>'
  strVar += '<b>Internet Explorer :</b>'
  strVar += 'vous devez peut-être répondre positivement à une invite éventuelle <b>en bas de la fenêtre</b>.'
  strVar += '</li>'
  strVar += '<li>'
  strVar += '<b>Sinon :</b>'
  strVar += 'Vous avez probablement désactivé le &laquo;javascript&raquo; (et pas &laquo;java&raquo;) sur votre'
  strVar += 'navigateur. Cherchez comment le réactiver.'
  strVar += '</li>'
  strVar += '</ul>'
  strVar += '</div>'
  strVar += '<div id="ALdimFenetre">'
  strVar += '<p>'
  strVar += 'Fenêtre graphique (hors tortue) :'
  strVar += '</p>'
  strVar += '<ul>'
  strVar += '<li>'
  strVar += 'x<sub>min</sub> : <input id="ALxMin" value="-5"/>'
  strVar += '</li>'
  strVar += '<li>'
  strVar += 'y<sub>min</sub> : <input id="ALyMin" value="-5"/>'
  strVar += '</li>'
  strVar += '<li>'
  strVar += 'x<sub>max</sub> : <input id="ALxMax" value="5"/>'
  strVar += '</li>'
  strVar += '<li>'
  strVar += 'y<sub>max</sub> : <input id="ALyMax" value="5"/>'
  strVar += '</li>'
  strVar += '<li>'
  strVar += 'Couleur de trait : <input id="ALstroke" value="noir"/>'
  strVar += '</li>'
  strVar += '<li>'
  strVar += '<input type="checkbox" id="ALaxes" />  Axes visibles<br>'
  strVar += '</li>'
  strVar += '</ul>'
  strVar += '<input type="button" onclick="ALdimGraph()" value="ok"/>'
  strVar += '</div>'
  divGauche.innerHTML = strVar

  // fin de l’injection code html  dans la page ----------------------------------

  // injection de l’algorithme passé en argument
  if (algo) {
    const d = document.getElementById('ALgauche')
    d.innerHTML = '<pre>\n' + algo + '\n</pre>'
    parseAlgo()
  }
  // désactivation de la correction automatique
  document.body.setAttribute('spellcheck', 'false')
  // activation de la sauvegarde d’urgence
  // document.body.setAttribute('onbeforeunload', 'ALsauveUrgence()')
  // masquer certains élements
  let elt
  for (var nom in this.aMasquer) {
    elt = document.getElementById(nom)
    if (elt) {
      elt.style.display = 'none'
      elt.style.visibility = 'hidden'
    }
  }
  // remplissage des démos
  elt = document.getElementById('ALdemos')
  let h1, pre, texte
  for (nom in this.demos) {
    h1 = document.createElement('h1')
    pre = document.createElement('pre')
    texte = document.createTextNode(nom)
    h1.appendChild(texte)
    texte = document.createTextNode(this.demos[nom])
    pre.appendChild(texte)
    elt.appendChild(h1)
    elt.appendChild(pre)
  }
  // précision
  document.getElementById('ALinputPrecis').value = this.precis
  // initialisations
  this.init() // ex ALinit
}

Psylvia.prototype.init = function init (algoInitial) {
  document.execCommand('enableObjectResizing', false, 'false')
  document.getElementById('ALavertissement').style.display = 'none'
  // document.getElementById("ALinputPrecis").value=w.ALprecision
  faitPrecision()
  ALafficheBalai('ALeffaceG')
  ALafficheBalai('ALeffaceD')
  initDemos()
  document.getElementById('ALselectDemo').selectedIndex = 0
  const g = document.getElementById('ALgauche')
  g.focus()
  Tortue.prototype.faitRepas()
  if (!algoInitial) {
    ALinsere('<pre>Taper un algorithme ici :<br></pre>')
  }
  g.blur()
  parseAlgo()
  modeTortue()
}