// 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(/<sup>/gi, '<sup>')
texte = texte.replace(/<\/sup>/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, '<sup>')
texte = texte.replace(/<\/sup>/gi, '</sup>')
for (let i = 0; i < expo.length; i++) {
reg = new RegExp(expo[i], 'gi')
s = '<sup>' + i + '</sup>'
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(/</gi, '<')
s = s.replace(/>/gi, '>')
s = s.replace(/≤/gi, '<=')
s = s.replace(/≥/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 = ['<', '>', "<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> </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>', '<', '>', '≤', '≥', '≠', 'π', '→', '&']
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 = ['<', '>', '≤', '≥', '≠', 'π', '→', '', '', '\\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('π')\" title=\"insère π\">π </button>\n"
strVar += "<button onclick=\"ALinsere('≤')\" title=\"insère ≤\">≤ </button>\n"
strVar += "<button onclick=\"ALinsere('≥')\" title=\"insère ≥\">≥ </button>\n"
strVar += "<button onclick=\"ALinsere('≠')\" title=\"insère ≠\">≠ </button>\n"
strVar += "<button onclick=\"ALinsere('→')\" title=\"insère →\">→ </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 «javascript» (et pas «java») 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()
}