| <html> |
| <head> |
| <div style="height:0"> |
| |
| <div id="cubic1"> |
| {{3.13,2.74}, {1.08,4.62}, {3.71,0.94}, {2.01,3.81}} |
| {{6.71,3.14}, {7.99,2.75}, {8.27,1.96}, {6.35,3.57}} |
| {{9.45,10.67}, {10.05,5.78}, {13.95,7.46}, {14.72,5.29}} |
| {{3.34,8.98}, {1.95,10.27}, {3.76,7.65}, {4.96,10.64}} |
| </div> |
| |
| </div> |
| |
| <script type="text/javascript"> |
| |
| var testDivs = [ |
| cubic1, |
| ]; |
| |
| var scale, columns, rows, xStart, yStart; |
| |
| var ticks = 10; |
| var at_x = 13 + 0.5; |
| var at_y = 23 + 0.5; |
| var decimal_places = 3; |
| var tests = []; |
| var testTitles = []; |
| var testIndex = 0; |
| var ctx; |
| var minScale = 1; |
| var subscale = 1; |
| var curveT = -1; |
| var xmin, xmax, ymin, ymax; |
| |
| var mouseX, mouseY; |
| var mouseDown = false; |
| |
| var draw_deriviatives = false; |
| var draw_endpoints = true; |
| var draw_hodo = false; |
| var draw_hodo2 = false; |
| var draw_hodo_origin = true; |
| var draw_midpoint = false; |
| var draw_tangents = true; |
| var draw_sequence = true; |
| |
| function parse(test, title) { |
| var curveStrs = test.split("{{"); |
| if (curveStrs.length == 1) |
| curveStrs = test.split("=("); |
| var pattern = /[a-z$=]?-?\d+\.*\d*e?-?\d*/g; |
| var curves = []; |
| for (var c in curveStrs) { |
| var curveStr = curveStrs[c]; |
| var points = curveStr.match(pattern); |
| var pts = []; |
| for (var wd in points) { |
| var num = parseFloat(points[wd]); |
| if (isNaN(num)) continue; |
| pts.push(num); |
| } |
| if (pts.length > 2) |
| curves.push(pts); |
| } |
| if (curves.length >= 1) { |
| tests.push(curves); |
| testTitles.push(title); |
| } |
| } |
| |
| function init(test) { |
| var canvas = document.getElementById('canvas'); |
| if (!canvas.getContext) return; |
| canvas.width = window.innerWidth - 20; |
| canvas.height = window.innerHeight - 20; |
| ctx = canvas.getContext('2d'); |
| xmin = Infinity; |
| xmax = -Infinity; |
| ymin = Infinity; |
| ymax = -Infinity; |
| for (var curves in test) { |
| var curve = test[curves]; |
| var last = curve.length; |
| for (var idx = 0; idx < last; idx += 2) { |
| xmin = Math.min(xmin, curve[idx]); |
| xmax = Math.max(xmax, curve[idx]); |
| ymin = Math.min(ymin, curve[idx + 1]); |
| ymax = Math.max(ymax, curve[idx + 1]); |
| } |
| } |
| xmin -= 1; |
| var testW = xmax - xmin; |
| var testH = ymax - ymin; |
| subscale = 1; |
| while (testW * subscale < 0.1 && testH * subscale < 0.1) { |
| subscale *= 10; |
| } |
| while (testW * subscale > 10 && testH * subscale > 10) { |
| subscale /= 10; |
| } |
| calcFromScale(); |
| } |
| |
| function hodograph(cubic) { |
| var hodo = []; |
| hodo[0] = 3 * (cubic[2] - cubic[0]); |
| hodo[1] = 3 * (cubic[3] - cubic[1]); |
| hodo[2] = 3 * (cubic[4] - cubic[2]); |
| hodo[3] = 3 * (cubic[5] - cubic[3]); |
| hodo[4] = 3 * (cubic[6] - cubic[4]); |
| hodo[5] = 3 * (cubic[7] - cubic[5]); |
| return hodo; |
| } |
| |
| function hodograph2(cubic) { |
| var quad = hodograph(cubic); |
| var hodo = []; |
| hodo[0] = 2 * (quad[2] - quad[0]); |
| hodo[1] = 2 * (quad[3] - quad[1]); |
| hodo[2] = 2 * (quad[4] - quad[2]); |
| hodo[3] = 2 * (quad[5] - quad[3]); |
| return hodo; |
| } |
| |
| function quadraticRootsReal(A, B, C, s) { |
| if (A == 0) { |
| if (B == 0) { |
| s[0] = 0; |
| return C == 0; |
| } |
| s[0] = -C / B; |
| return 1; |
| } |
| /* normal form: x^2 + px + q = 0 */ |
| var p = B / (2 * A); |
| var q = C / A; |
| var p2 = p * p; |
| if (p2 < q) { |
| return 0; |
| } |
| var sqrt_D = 0; |
| if (p2 > q) { |
| sqrt_D = sqrt(p2 - q); |
| } |
| s[0] = sqrt_D - p; |
| s[1] = -sqrt_D - p; |
| return 1 + s[0] != s[1]; |
| } |
| |
| function add_valid_ts(s, realRoots, t) { |
| var foundRoots = 0; |
| for (var index = 0; index < realRoots; ++index) { |
| var tValue = s[index]; |
| if (tValue >= 0 && tValue <= 1) { |
| for (var idx2 = 0; idx2 < foundRoots; ++idx2) { |
| if (t[idx2] != tValue) { |
| t[foundRoots++] = tValue; |
| } |
| } |
| } |
| } |
| return foundRoots; |
| } |
| |
| function quadraticRootsValidT(a, b, c, t) { |
| var s = []; |
| var realRoots = quadraticRootsReal(A, B, C, s); |
| var foundRoots = add_valid_ts(s, realRoots, t); |
| return foundRoots != 0; |
| } |
| |
| function find_cubic_inflections(cubic, tValues) |
| { |
| var Ax = src[2] - src[0]; |
| var Ay = src[3] - src[1]; |
| var Bx = src[4] - 2 * src[2] + src[0]; |
| var By = src[5] - 2 * src[3] + src[1]; |
| var Cx = src[6] + 3 * (src[2] - src[4]) - src[0]; |
| var Cy = src[7] + 3 * (src[3] - src[5]) - src[1]; |
| return quadraticRootsValidT(Bx * Cy - By * Cx, (Ax * Cy - Ay * Cx), |
| Ax * By - Ay * Bx, tValues); |
| } |
| |
| function dx_at_t(cubic, t) { |
| var one_t = 1 - t; |
| var a = cubic[0]; |
| var b = cubic[2]; |
| var c = cubic[4]; |
| var d = cubic[6]; |
| return 3 * ((b - a) * one_t * one_t + 2 * (c - b) * t * one_t + (d - c) * t * t); |
| } |
| |
| function dy_at_t(cubic, t) { |
| var one_t = 1 - t; |
| var a = cubic[1]; |
| var b = cubic[3]; |
| var c = cubic[5]; |
| var d = cubic[7]; |
| return 3 * ((b - a) * one_t * one_t + 2 * (c - b) * t * one_t + (d - c) * t * t); |
| } |
| |
| function x_at_t(cubic, t) { |
| var one_t = 1 - t; |
| var one_t2 = one_t * one_t; |
| var a = one_t2 * one_t; |
| var b = 3 * one_t2 * t; |
| var t2 = t * t; |
| var c = 3 * one_t * t2; |
| var d = t2 * t; |
| return a * cubic[0] + b * cubic[2] + c * cubic[4] + d * cubic[6]; |
| } |
| |
| function y_at_t(cubic, t) { |
| var one_t = 1 - t; |
| var one_t2 = one_t * one_t; |
| var a = one_t2 * one_t; |
| var b = 3 * one_t2 * t; |
| var t2 = t * t; |
| var c = 3 * one_t * t2; |
| var d = t2 * t; |
| return a * cubic[1] + b * cubic[3] + c * cubic[5] + d * cubic[7]; |
| } |
| |
| function calcFromScale() { |
| xStart = Math.floor(xmin * subscale) / subscale; |
| yStart = Math.floor(ymin * subscale) / subscale; |
| var xEnd = Math.ceil(xmin * subscale) / subscale; |
| var yEnd = Math.ceil(ymin * subscale) / subscale; |
| var cCelsW = Math.floor(ctx.canvas.width / 10); |
| var cCelsH = Math.floor(ctx.canvas.height / 10); |
| var testW = xEnd - xStart; |
| var testH = yEnd - yStart; |
| var scaleWH = 1; |
| while (cCelsW > testW * scaleWH * 10 && cCelsH > testH * scaleWH * 10) { |
| scaleWH *= 10; |
| } |
| while (cCelsW * 10 < testW * scaleWH && cCelsH * 10 < testH * scaleWH) { |
| scaleWH /= 10; |
| } |
| |
| columns = Math.ceil(xmax * subscale) - Math.floor(xmin * subscale) + 1; |
| rows = Math.ceil(ymax * subscale) - Math.floor(ymin * subscale) + 1; |
| |
| var hscale = ctx.canvas.width / columns / ticks; |
| var vscale = ctx.canvas.height / rows / ticks; |
| minScale = Math.floor(Math.min(hscale, vscale)); |
| scale = minScale * subscale; |
| } |
| |
| function drawLine(x1, y1, x2, y2) { |
| var unit = scale * ticks; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| ctx.beginPath(); |
| ctx.moveTo(xoffset + x1 * unit, yoffset + y1 * unit); |
| ctx.lineTo(xoffset + x2 * unit, yoffset + y2 * unit); |
| ctx.stroke(); |
| } |
| |
| function drawPoint(px, py) { |
| var unit = scale * ticks; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| var _px = px * unit + xoffset; |
| var _py = py * unit + yoffset; |
| ctx.beginPath(); |
| ctx.arc(_px, _py, 3, 0, Math.PI*2, true); |
| ctx.closePath(); |
| ctx.stroke(); |
| } |
| |
| function drawPointSolid(px, py) { |
| drawPoint(px, py); |
| ctx.fillStyle = "rgba(0,0,0, 0.4)"; |
| ctx.fill(); |
| } |
| |
| function drawLabel(num, px, py) { |
| ctx.beginPath(); |
| ctx.arc(px, py, 8, 0, Math.PI*2, true); |
| ctx.closePath(); |
| ctx.strokeStyle = "rgba(0,0,0, 0.4)"; |
| ctx.lineWidth = num == 0 || num == 3 ? 2 : 1; |
| ctx.stroke(); |
| ctx.fillStyle = "black"; |
| ctx.font = "normal 10px Arial"; |
| // ctx.rotate(0.001); |
| ctx.fillText(num, px - 2, py + 3); |
| // ctx.rotate(-0.001); |
| } |
| |
| function drawLabelX(ymin, num, loc) { |
| var unit = scale * ticks; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| var px = loc * unit + xoffset; |
| var py = ymin * unit + yoffset - 20; |
| drawLabel(num, px, py); |
| } |
| |
| function drawLabelY(xmin, num, loc) { |
| var unit = scale * ticks; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| var px = xmin * unit + xoffset - 20; |
| var py = loc * unit + yoffset; |
| drawLabel(num, px, py); |
| } |
| |
| function drawHodoOrigin(hx, hy, hMinX, hMinY, hMaxX, hMaxY) { |
| ctx.beginPath(); |
| ctx.moveTo(hx, hy - 100); |
| ctx.lineTo(hx, hy); |
| ctx.strokeStyle = hMinY < 0 ? "green" : "blue"; |
| ctx.stroke(); |
| ctx.beginPath(); |
| ctx.moveTo(hx, hy); |
| ctx.lineTo(hx, hy + 100); |
| ctx.strokeStyle = hMaxY > 0 ? "green" : "blue"; |
| ctx.stroke(); |
| ctx.beginPath(); |
| ctx.moveTo(hx - 100, hy); |
| ctx.lineTo(hx, hy); |
| ctx.strokeStyle = hMinX < 0 ? "green" : "blue"; |
| ctx.stroke(); |
| ctx.beginPath(); |
| ctx.moveTo(hx, hy); |
| ctx.lineTo(hx + 100, hy); |
| ctx.strokeStyle = hMaxX > 0 ? "green" : "blue"; |
| ctx.stroke(); |
| } |
| |
| function logCurves(test) { |
| for (curves in test) { |
| var curve = test[curves]; |
| if (curve.length != 8) { |
| continue; |
| } |
| var str = "{{"; |
| for (i = 0; i < 8; i += 2) { |
| str += curve[i].toFixed(2) + "," + curve[i + 1].toFixed(2); |
| if (i < 6) { |
| str += "}, {"; |
| } |
| } |
| str += "}}"; |
| console.log(str); |
| } |
| } |
| |
| function scalexy(x, y, mag) { |
| var length = Math.sqrt(x * x + y * y); |
| return mag / length; |
| } |
| |
| function drawArrow(x, y, dx, dy) { |
| var unit = scale * ticks; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| var dscale = scalexy(dx, dy, 1); |
| dx *= dscale; |
| dy *= dscale; |
| ctx.beginPath(); |
| ctx.moveTo(xoffset + x * unit, yoffset + y * unit); |
| x += dx; |
| y += dy; |
| ctx.lineTo(xoffset + x * unit, yoffset + y * unit); |
| dx /= 10; |
| dy /= 10; |
| ctx.lineTo(xoffset + (x - dy) * unit, yoffset + (y + dx) * unit); |
| ctx.lineTo(xoffset + (x + dx * 2) * unit, yoffset + (y + dy * 2) * unit); |
| ctx.lineTo(xoffset + (x + dy) * unit, yoffset + (y - dx) * unit); |
| ctx.lineTo(xoffset + x * unit, yoffset + y * unit); |
| ctx.strokeStyle = "rgba(0,75,0, 0.4)"; |
| ctx.stroke(); |
| } |
| |
| function draw(test, title) { |
| ctx.fillStyle = "rgba(0,0,0, 0.1)"; |
| ctx.font = "normal 50px Arial"; |
| ctx.fillText(title, 50, 50); |
| ctx.font = "normal 10px Arial"; |
| var unit = scale * ticks; |
| // ctx.lineWidth = "1.001"; "0.999"; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| |
| for (curves in test) { |
| var curve = test[curves]; |
| if (curve.length != 8) { |
| continue; |
| } |
| ctx.lineWidth = 1; |
| if (draw_tangents) { |
| ctx.strokeStyle = "rgba(0,0,255, 0.3)"; |
| drawLine(curve[0], curve[1], curve[2], curve[3]); |
| drawLine(curve[2], curve[3], curve[4], curve[5]); |
| drawLine(curve[4], curve[5], curve[6], curve[7]); |
| } |
| if (draw_deriviatives) { |
| var dx = dx_at_t(curve, 0); |
| var dy = dy_at_t(curve, 0); |
| drawArrow(curve[0], curve[1], dx, dy); |
| dx = dx_at_t(curve, 1); |
| dy = dy_at_t(curve, 1); |
| drawArrow(curve[6], curve[7], dx, dy); |
| if (draw_midpoint) { |
| var midX = x_at_t(curve, 0.5); |
| var midY = y_at_t(curve, 0.5); |
| dx = dx_at_t(curve, 0.5); |
| dy = dy_at_t(curve, 0.5); |
| drawArrow(midX, midY, dx, dy); |
| } |
| } |
| ctx.beginPath(); |
| ctx.moveTo(xoffset + curve[0] * unit, yoffset + curve[1] * unit); |
| ctx.bezierCurveTo( |
| xoffset + curve[2] * unit, yoffset + curve[3] * unit, |
| xoffset + curve[4] * unit, yoffset + curve[5] * unit, |
| xoffset + curve[6] * unit, yoffset + curve[7] * unit); |
| ctx.strokeStyle = "black"; |
| ctx.stroke(); |
| if (draw_endpoints) { |
| drawPoint(curve[0], curve[1]); |
| drawPoint(curve[2], curve[3]); |
| drawPoint(curve[4], curve[5]); |
| drawPoint(curve[6], curve[7]); |
| } |
| if (draw_midpoint) { |
| var midX = x_at_t(curve, 0.5); |
| var midY = y_at_t(curve, 0.5); |
| drawPointSolid(midX, midY); |
| } |
| if (draw_hodo) { |
| var hodo = hodograph(curve); |
| var hMinX = Math.min(0, hodo[0], hodo[2], hodo[4]); |
| var hMinY = Math.min(0, hodo[1], hodo[3], hodo[5]); |
| var hMaxX = Math.max(0, hodo[0], hodo[2], hodo[4]); |
| var hMaxY = Math.max(0, hodo[1], hodo[3], hodo[5]); |
| var hScaleX = hMaxX - hMinX > 0 ? ctx.canvas.width / (hMaxX - hMinX) : 1; |
| var hScaleY = hMaxY - hMinY > 0 ? ctx.canvas.height / (hMaxY - hMinY) : 1; |
| var hUnit = Math.min(hScaleX, hScaleY); |
| hUnit /= 2; |
| var hx = xoffset - hMinX * hUnit; |
| var hy = yoffset - hMinY * hUnit; |
| ctx.moveTo(hx + hodo[0] * hUnit, hy + hodo[1] * hUnit); |
| ctx.quadraticCurveTo( |
| hx + hodo[2] * hUnit, hy + hodo[3] * hUnit, |
| hx + hodo[4] * hUnit, hy + hodo[5] * hUnit); |
| ctx.strokeStyle = "red"; |
| ctx.stroke(); |
| if (draw_hodo_origin) { |
| drawHodoOrigin(hx, hy, hMinX, hMinY, hMaxX, hMaxY); |
| } |
| } |
| if (draw_hodo2) { |
| var hodo = hodograph2(curve); |
| var hMinX = Math.min(0, hodo[0], hodo[2]); |
| var hMinY = Math.min(0, hodo[1], hodo[3]); |
| var hMaxX = Math.max(0, hodo[0], hodo[2]); |
| var hMaxY = Math.max(0, hodo[1], hodo[3]); |
| var hScaleX = hMaxX - hMinX > 0 ? ctx.canvas.width / (hMaxX - hMinX) : 1; |
| var hScaleY = hMaxY - hMinY > 0 ? ctx.canvas.height / (hMaxY - hMinY) : 1; |
| var hUnit = Math.min(hScaleX, hScaleY); |
| hUnit /= 2; |
| var hx = xoffset - hMinX * hUnit; |
| var hy = yoffset - hMinY * hUnit; |
| ctx.moveTo(hx + hodo[0] * hUnit, hy + hodo[1] * hUnit); |
| ctx.lineTo(hx + hodo[2] * hUnit, hy + hodo[3] * hUnit); |
| ctx.strokeStyle = "red"; |
| ctx.stroke(); |
| drawHodoOrigin(hx, hy, hMinX, hMinY, hMaxX, hMaxY); |
| } |
| if (draw_sequence) { |
| var ymin = Math.min(curve[1], curve[3], curve[5], curve[7]); |
| for (var i = 0; i < 8; i+= 2) { |
| drawLabelX(ymin, i >> 1, curve[i]); |
| } |
| var xmin = Math.min(curve[0], curve[2], curve[4], curve[6]); |
| for (var i = 1; i < 8; i+= 2) { |
| drawLabelY(xmin, i >> 1, curve[i]); |
| } |
| } |
| } |
| } |
| |
| function drawTop() { |
| init(tests[testIndex]); |
| redraw(); |
| } |
| |
| function redraw() { |
| ctx.beginPath(); |
| ctx.rect(0, 0, ctx.canvas.width, ctx.canvas.height); |
| ctx.fillStyle="white"; |
| ctx.fill(); |
| draw(tests[testIndex], testTitles[testIndex]); |
| } |
| |
| function doKeyPress(evt) { |
| var char = String.fromCharCode(evt.charCode); |
| switch (char) { |
| case '2': |
| draw_hodo2 ^= true; |
| redraw(); |
| break; |
| case 'd': |
| draw_deriviatives ^= true; |
| redraw(); |
| break; |
| case 'e': |
| draw_endpoints ^= true; |
| redraw(); |
| break; |
| case 'h': |
| draw_hodo ^= true; |
| redraw(); |
| break; |
| case 'N': |
| testIndex += 9; |
| case 'n': |
| if (++testIndex >= tests.length) |
| testIndex = 0; |
| drawTop(); |
| break; |
| case 'l': |
| logCurves(tests[testIndex]); |
| break; |
| case 'm': |
| draw_midpoint ^= true; |
| redraw(); |
| break; |
| case 'o': |
| draw_hodo_origin ^= true; |
| redraw(); |
| break; |
| case 'P': |
| testIndex -= 9; |
| case 'p': |
| if (--testIndex < 0) |
| testIndex = tests.length - 1; |
| drawTop(); |
| break; |
| case 's': |
| draw_sequence ^= true; |
| redraw(); |
| break; |
| case 't': |
| draw_tangents ^= true; |
| redraw(); |
| break; |
| } |
| } |
| |
| function calcXY() { |
| var e = window.event; |
| var tgt = e.target || e.srcElement; |
| var left = tgt.offsetLeft; |
| var top = tgt.offsetTop; |
| var unit = scale * ticks; |
| mouseX = (e.clientX - left - Math.ceil(at_x) + 1) / unit + xStart; |
| mouseY = (e.clientY - top - Math.ceil(at_y)) / unit + yStart; |
| } |
| |
| var lastX, lastY; |
| var activeCurve = []; |
| var activePt; |
| |
| function handleMouseClick() { |
| calcXY(); |
| } |
| |
| function initDown() { |
| var unit = scale * ticks; |
| var xoffset = xStart * -unit + at_x; |
| var yoffset = yStart * -unit + at_y; |
| var test = tests[testIndex]; |
| var bestDistance = 1000000; |
| activePt = -1; |
| for (curves in test) { |
| var testCurve = test[curves]; |
| if (testCurve.length != 8) { |
| continue; |
| } |
| for (var i = 0; i < 8; i += 2) { |
| var testX = testCurve[i]; |
| var testY = testCurve[i + 1]; |
| var dx = testX - mouseX; |
| var dy = testY - mouseY; |
| var dist = dx * dx + dy * dy; |
| if (dist > bestDistance) { |
| continue; |
| } |
| activeCurve = testCurve; |
| activePt = i; |
| bestDistance = dist; |
| } |
| } |
| if (activePt >= 0) { |
| lastX = mouseX; |
| lastY = mouseY; |
| } |
| } |
| |
| function handleMouseOver() { |
| if (!mouseDown) { |
| activePt = -1; |
| return; |
| } |
| calcXY(); |
| if (activePt < 0) { |
| initDown(); |
| return; |
| } |
| var unit = scale * ticks; |
| var deltaX = (mouseX - lastX) /* / unit */; |
| var deltaY = (mouseY - lastY) /*/ unit */; |
| lastX = mouseX; |
| lastY = mouseY; |
| activeCurve[activePt] += deltaX; |
| activeCurve[activePt + 1] += deltaY; |
| redraw(); |
| } |
| |
| function start() { |
| for (i = 0; i < testDivs.length; ++i) { |
| var title = testDivs[i].id.toString(); |
| var str = testDivs[i].firstChild.data; |
| parse(str, title); |
| } |
| drawTop(); |
| window.addEventListener('keypress', doKeyPress, true); |
| window.onresize = function() { |
| drawTop(); |
| } |
| } |
| |
| </script> |
| </head> |
| |
| <body onLoad="start();"> |
| <canvas id="canvas" width="750" height="500" |
| onmousedown="mouseDown = true" |
| onmouseup="mouseDown = false" |
| onmousemove="handleMouseOver()" |
| onclick="handleMouseClick()" |
| ></canvas > |
| </body> |
| </html> |