2023-08-30 03:56:13 +00:00
|
|
|
import https://code.jquery.com/jquery-3.7.1.js
|
2023-09-11 01:52:39 +00:00
|
|
|
import type {AppletObject} from ./deps/geogebra/api.ts
|
2023-08-30 03:56:13 +00:00
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
type AppletDescription
|
|
|
|
html: string
|
|
|
|
children: HTMLCollection
|
|
|
|
id: string
|
|
|
|
width: number
|
|
|
|
height: number
|
|
|
|
|
|
|
|
joyceApplets: AppletDescription[] := []
|
2023-08-30 03:56:13 +00:00
|
|
|
$('applet[code="Geometry"]').before (i, html) ->
|
|
|
|
id := `joyceApplet${i}`
|
|
|
|
joyceApplets.push { html, this.children, id,
|
2023-09-11 01:52:39 +00:00
|
|
|
width: parseInt(this.getAttribute('width') ?? '200'),
|
|
|
|
height: parseInt(this.getAttribute('height') ?? '200') }
|
2023-08-30 03:56:13 +00:00
|
|
|
`<div id="${id}"></div>`
|
|
|
|
|
|
|
|
jQuery.getScript 'https://www.geogebra.org/apps/deployggb.js', =>
|
|
|
|
for each jApp of joyceApplets
|
|
|
|
params := {
|
|
|
|
appName: 'classic',
|
|
|
|
jApp.width,
|
|
|
|
jApp.height,
|
2023-09-11 01:52:39 +00:00
|
|
|
appletOnLoad: (api: AppletObject) =>
|
2023-08-30 03:56:13 +00:00
|
|
|
for child of jApp.children
|
|
|
|
dispatchJcommand api, child
|
|
|
|
api.setCoordSystem(-10, 10 + jApp.width, -10, 10 + jApp.height)
|
2023-09-11 01:52:39 +00:00
|
|
|
} as const
|
2023-08-30 03:56:13 +00:00
|
|
|
geoApp := new GGBApplet params
|
|
|
|
geoApp.inject jApp.id
|
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
type GeogebraCallback = (api: AppletObject) => void
|
|
|
|
type Commander
|
|
|
|
command: string
|
|
|
|
callbacks: GeogebraCallback[]
|
|
|
|
|
|
|
|
type ClassHandler = (
|
|
|
|
name: string, m: string, data: string, colors: string[]) => Commander
|
2023-08-30 03:56:13 +00:00
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
function dispatchJcommand(api: AppletObject, param: Element): void
|
2023-08-30 03:56:13 +00:00
|
|
|
val := param.getAttribute 'value'
|
2023-09-11 01:52:39 +00:00
|
|
|
unless val return
|
2023-08-30 03:56:13 +00:00
|
|
|
switch param.getAttribute 'name'
|
|
|
|
'background'
|
|
|
|
api.setGraphicsOptions 1, bgColor: `#${val}`
|
|
|
|
'title'
|
|
|
|
api.evalCommand `TitlePoint = Corner(1,1)
|
|
|
|
Text("${val}", TitlePoint + (2,5))`
|
|
|
|
/e\[\d+\]/
|
|
|
|
{command, callbacks} := jToG val
|
|
|
|
if command
|
|
|
|
if api.evalCommand command
|
|
|
|
for each cb of callbacks
|
|
|
|
cb(api)
|
|
|
|
else
|
|
|
|
console.log `Geogebra command '${command}' translated from`,
|
|
|
|
val, 'failed.'
|
|
|
|
else
|
2023-09-19 07:32:11 +00:00
|
|
|
console.log `Could not parse command '${val}'`
|
2023-08-30 03:56:13 +00:00
|
|
|
else
|
|
|
|
console.log `Unkown param ${param}`
|
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
function jToG(jCom: string): Commander
|
2023-08-30 03:56:13 +00:00
|
|
|
[name, klass, method, data, ...colors] := jCom.split(';')
|
|
|
|
if klass in classHandler
|
|
|
|
return classHandler[klass] name, method, data, colors
|
|
|
|
console.log `Unknown entity class ${klass}`
|
2023-09-11 01:52:39 +00:00
|
|
|
command: '', callbacks: []
|
2023-08-30 03:56:13 +00:00
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
classHandler: Record<string, ClassHandler> :=
|
|
|
|
point: (name, method, data, colors) =>
|
2023-08-30 03:56:13 +00:00
|
|
|
command .= ''
|
2023-09-11 01:52:39 +00:00
|
|
|
callbacks: GeogebraCallback[] .= []
|
2023-09-19 07:32:11 +00:00
|
|
|
args := data.split(',')
|
2023-09-11 01:52:39 +00:00
|
|
|
switch method
|
2023-08-30 03:56:13 +00:00
|
|
|
/free|fixed/
|
|
|
|
command += `${name} = (${data})`
|
2023-09-11 01:52:39 +00:00
|
|
|
if method is 'fixed'
|
|
|
|
callbacks.push (api: AppletObject) => api.setFixed(name, true)
|
2023-08-30 03:56:13 +00:00
|
|
|
'perpendicular'
|
2023-09-19 07:32:11 +00:00
|
|
|
[center, direction] := args
|
2023-08-30 03:56:13 +00:00
|
|
|
command += `${name} = Rotate(${direction}, 3*pi/2, ${center})`
|
2023-09-19 07:32:11 +00:00
|
|
|
'angleDivider'
|
|
|
|
n .= -1
|
|
|
|
// use the fact that NaN doesn't equal itself:
|
|
|
|
nLoc := args.findIndex((arg) => (n = parseInt arg) is n)
|
|
|
|
if n >= 0
|
|
|
|
args.splice(nLoc)
|
|
|
|
[center, start, end] := args
|
|
|
|
command += `${name}aUx1 = Segment(${start}, ${end})
|
|
|
|
${name}aUx2 = Angle(${start}, ${center}, ${end})
|
|
|
|
${name}aUx2a = If(${name}aUx2 > pi, ${name}aUx2 - 2*pi, ${name}aUx2)
|
|
|
|
${name}aUx3 = Rotate(${start}, ${name}aUx2a/${n}, ${center})
|
|
|
|
${name}aUx4 = Ray(${center}, ${name}aUx3)
|
|
|
|
${name} = Intersect(${name}aUx1, ${name}aUx4)`
|
|
|
|
'intersection'
|
|
|
|
command += `${name} = Intersect(${data})`
|
2023-08-30 03:56:13 +00:00
|
|
|
return {command, callbacks}
|
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
line: (name, method, data, colors) =>
|
2023-08-30 03:56:13 +00:00
|
|
|
command .= ''
|
2023-09-11 01:52:39 +00:00
|
|
|
callbacks: GeogebraCallback[] .= []
|
2023-09-19 07:32:11 +00:00
|
|
|
args := data.split(',')
|
2023-09-11 01:52:39 +00:00
|
|
|
switch method
|
2023-08-30 03:56:13 +00:00
|
|
|
'connect'
|
|
|
|
command += `${name} = Segment(${data})`
|
2023-09-19 07:32:11 +00:00
|
|
|
'parallel'
|
|
|
|
[newStart, oldStart, oldEnd] := args
|
|
|
|
command += `${name}aUx1 = Vector(${oldStart}, ${newStart})
|
|
|
|
${name}aUx2 = Translate(${oldEnd}, ${name}aUx1)
|
|
|
|
${name} = Segment(${newStart}, ${name}aUx2)`
|
2023-08-30 03:56:13 +00:00
|
|
|
return {command, callbacks}
|
|
|
|
|
2023-09-11 01:52:39 +00:00
|
|
|
circle: (name, method, data, colors) =>
|
2023-08-30 03:56:13 +00:00
|
|
|
command .= ''
|
2023-09-11 01:52:39 +00:00
|
|
|
callbacks: GeogebraCallback[] .= []
|
|
|
|
switch method
|
2023-08-30 03:56:13 +00:00
|
|
|
'radius'
|
|
|
|
[center, point] := data.split(',')
|
|
|
|
command += `${name} = Circle(${center}, ${point})`
|
|
|
|
return {command, callbacks}
|