From d9d2de8544cd8dd94083bb7fb14f78edf1d28bf0 Mon Sep 17 00:00:00 2001 From: Glenn Rice Date: Thu, 12 Dec 2024 05:50:46 -0600 Subject: [PATCH] Make changes for existing graph tools that were made with the sine wave tool. For all of the graph tools switch to using the highlight point coordinates when a click occurs at an invalid location on the board for the next phase. Still refuse to create a point if a click or touch occurs off the board. This doesn't really happen for a click, but does for a touch on a touch screen device. In that case, the touch off the board should be ignored. Don't use `==` for comparison of floating point numbers. --- htdocs/js/GraphTool/cubictool.js | 43 ++++++++-------- htdocs/js/GraphTool/graphtool.js | 74 +++++++++++++++++----------- htdocs/js/GraphTool/intervaltools.js | 12 ++--- htdocs/js/GraphTool/quadratictool.js | 26 +++++----- htdocs/js/GraphTool/sinewavetool.js | 24 +++++---- 5 files changed, 96 insertions(+), 83 deletions(-) diff --git a/htdocs/js/GraphTool/cubictool.js b/htdocs/js/GraphTool/cubictool.js index 622844b16..d7d704045 100644 --- a/htdocs/js/GraphTool/cubictool.js +++ b/htdocs/js/GraphTool/cubictool.js @@ -240,13 +240,12 @@ }; this.phase2 = (coords) => { - // Don't allow the second point to be created on the same - // vertical line as the first point or off the board. - if ( - this.point1.X() == gt.snapRound(coords[1], gt.snapSizeX) || - !gt.boardHasPoint(coords[1], coords[2]) - ) - return; + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same vertical line as the first point, + // then use the highlight point coordinates instead. + if (Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps) + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); @@ -272,14 +271,15 @@ }; this.phase3 = (coords) => { - // Don't allow the third point to be created on the same vertical line as the - // first point, on the same vertical line as the second point, or off the board. + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same vertical line as the first point, or on the same + // vertical line as the second point, then use the highlight point coordinates instead. if ( - this.point1.X() == gt.snapRound(coords[1], gt.snapSizeX) || - this.point2.X() == gt.snapRound(coords[1], gt.snapSizeX) || - !gt.boardHasPoint(coords[1], coords[2]) + Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps || + Math.abs(this.point2.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps ) - return; + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); this.point3 = gt.graphObjectTypes.cubic.createPoint(coords[1], coords[2], [ @@ -311,16 +311,17 @@ }; this.phase4 = (coords) => { - // Don't allow the fourth point to be created on the same vertical line as the first - // point, on the same vertical line as the second point, on the same vertical line as - // the third point, or off the board. + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same vertical line as the first point, on the same vertical + // line as the second point, or on the same vertical line as the third point, then use the highlight + // point coordinates instead. if ( - this.point1.X() == gt.snapRound(coords[1], gt.snapSizeX) || - this.point2.X() == gt.snapRound(coords[1], gt.snapSizeX) || - this.point3.X() == gt.snapRound(coords[1], gt.snapSizeX) || - !gt.boardHasPoint(coords[1], coords[2]) + Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps || + Math.abs(this.point2.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps || + Math.abs(this.point3.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps ) - return; + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); diff --git a/htdocs/js/GraphTool/graphtool.js b/htdocs/js/GraphTool/graphtool.js index 06eb1c2ff..0474dcea8 100644 --- a/htdocs/js/GraphTool/graphtool.js +++ b/htdocs/js/GraphTool/graphtool.js @@ -686,7 +686,8 @@ window.graphTool = (containerId, options) => { // degenerate. gt.adjustDragPosition = (e, point, pairedPoint) => { if ( - (point.X() == pairedPoint?.X() && point.Y() == pairedPoint?.Y()) || + (Math.abs(point.X() - pairedPoint?.X()) < JXG.Math.eps && + Math.abs(point.Y() - pairedPoint?.Y()) < JXG.Math.eps) || !gt.boardHasPoint(point.X(), point.Y()) ) { const bbox = gt.board.getBoundingBox(); @@ -696,7 +697,11 @@ window.graphTool = (containerId, options) => { let y = point.Y() < bbox[3] ? bbox[3] : point.Y() > bbox[1] ? bbox[1] : point.Y(); // Adjust position of the point if it has the same coordinates as its paired point. - if (pairedPoint && x === pairedPoint.X() && y === pairedPoint.Y()) { + if ( + pairedPoint && + Math.abs(x - pairedPoint.X()) < JXG.Math.eps && + Math.abs(y - pairedPoint.Y()) < JXG.Math.eps + ) { let xDir, yDir; if (e.type === 'pointermove') { @@ -735,7 +740,11 @@ window.graphTool = (containerId, options) => { // horizontal or vertical line as its paired point by a drag. Note that when this method is called, the point has // already been moved by JSXGraph. This prevents parabolas from being made degenerate. gt.adjustDragPositionRestricted = (e, point, pairedPoint) => { - if (point.X() == pairedPoint?.X() || point.Y() == pairedPoint?.Y() || !gt.boardHasPoint(point.X(), point.Y())) { + if ( + Math.abs(point.X() - pairedPoint?.X()) < JXG.Math.eps || + Math.abs(point.Y() - pairedPoint?.Y()) < JXG.Math.eps || + !gt.boardHasPoint(point.X(), point.Y()) + ) { const bbox = gt.board.getBoundingBox(); // Clamp the coordinates to the board. @@ -756,8 +765,8 @@ window.graphTool = (containerId, options) => { yDir = e.key === 'ArrowUp' ? 1 : e.key === 'ArrowDown' ? -1 : 0; } - if (x == pairedPoint.X()) x += xDir * gt.snapSizeX; - if (y == pairedPoint.Y()) y += yDir * gt.snapSizeY; + if (Math.abs(x - pairedPoint.X()) < JXG.Math.eps) x += xDir * gt.snapSizeX; + if (Math.abs(y - pairedPoint.Y()) < JXG.Math.eps) y += yDir * gt.snapSizeY; // If the computed new coordinates are off the board, // then move the coordinates the other direction instead. @@ -1524,8 +1533,9 @@ window.graphTool = (containerId, options) => { // object's defining points. for (const point of gt.selectedObj.definingPts) { if ( - point.X() == gt.snapRound(coords.usrCoords[1], gt.snapSizeX) && - point.Y() == gt.snapRound(coords.usrCoords[2], gt.snapSizeY) + Math.abs(point.X() - gt.snapRound(coords.usrCoords[1], gt.snapSizeX)) < + JXG.Math.eps && + Math.abs(point.Y() - gt.snapRound(coords.usrCoords[2], gt.snapSizeY)) < JXG.Math.eps ) return; } @@ -1735,16 +1745,17 @@ window.graphTool = (containerId, options) => { gt.board.update(); } - // In phase2 the user has selected a second point. If that point is on the board - // and is not the same as the first point, then finalize the line. + // In phase2 the user has selected a second point. If that point is on the board , then finalize the line. phase2(coords) { - // Don't allow the second point to be created on top of the first or off the board + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are the same those of the first point, + // then use the highlight point coordinates instead. if ( - (this.point1.X() == gt.snapRound(coords[1], gt.snapSizeX) && - this.point1.Y() == gt.snapRound(coords[2], gt.snapSizeY)) || - !gt.boardHasPoint(coords[1], coords[2]) + Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps && + Math.abs(this.point1.Y() - gt.snapRound(coords[2], gt.snapSizeY)) < JXG.Math.eps ) - return; + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); @@ -1848,11 +1859,11 @@ window.graphTool = (containerId, options) => { gt.board.on('up', (e) => this.phase1(gt.getMouseCoords(e).usrCoords)); } - // In phase1 the user has selected a point. If that point is on the board, then make - // that the center of the circle, and set up phase2. + // In phase1 the user has selected a point. If the point is on the board, then create the center of the circle, + // and set up phase2. phase1(coords) { - // Don't allow the point to be created off the board. if (!gt.boardHasPoint(coords[1], coords[2])) return; + gt.board.off('up'); this.center = gt.board.create('point', [coords[1], coords[2]], { @@ -1880,16 +1891,17 @@ window.graphTool = (containerId, options) => { gt.board.update(); } - // In phase2 the user has selected a second point. If that point is on the board - // and is not the same as the center, then finalize the circle. + // In phase2 the user has selected a second point. If that point is on the board, then finalize the circle. phase2(coords) { - // Don't allow the second point to be created on top of the center or off the board + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are the same those of the first point, + // then use the highlight point coordinates instead. if ( - (this.center.X() == gt.snapRound(coords[1], gt.snapSizeX) && - this.center.Y() == gt.snapRound(coords[2], gt.snapSizeY)) || - !gt.boardHasPoint(coords[1], coords[2]) + Math.abs(this.center.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps && + Math.abs(this.center.Y() - gt.snapRound(coords[2], gt.snapSizeY)) < JXG.Math.eps ) - return; + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); @@ -2041,14 +2053,15 @@ window.graphTool = (containerId, options) => { } phase2(coords) { - // Don't allow the second point to be created on the same - // horizontal or vertical line as the vertex or off the board. + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same horizontal or vertical line as the vertex, + // then use the highlight point coordinates instead. if ( - this.vertex.X() == gt.snapRound(coords[1], gt.snapSizeX) || - this.vertex.Y() == gt.snapRound(coords[2], gt.snapSizeY) || - !gt.boardHasPoint(coords[1], coords[2]) + Math.abs(this.vertex.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps || + Math.abs(this.vertex.Y() - gt.snapRound(coords[2], gt.snapSizeY)) < JXG.Math.eps ) - return; + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); @@ -2175,6 +2188,7 @@ window.graphTool = (containerId, options) => { phase1(coords) { // Don't allow the fill to be created off the board if (!gt.boardHasPoint(coords[1], coords[2])) return; + gt.board.off('up'); gt.selectedObj = new gt.graphObjectTypes.fill(gt.createPoint(coords[1], coords[2])); diff --git a/htdocs/js/GraphTool/intervaltools.js b/htdocs/js/GraphTool/intervaltools.js index 26e07a872..d6c154ebd 100644 --- a/htdocs/js/GraphTool/intervaltools.js +++ b/htdocs/js/GraphTool/intervaltools.js @@ -559,12 +559,12 @@ }; this.phase2 = (coords) => { - // Don't allow the second point to be created on the first point or off the board. - if ( - this.point1.X() === gt.snapRound(coords[1], gt.snapSizeX) || - !gt.boardHasPoint(coords[1], coords[2]) - ) - return; + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are the same as those of the first point, + // then use the highlight point coordinates instead. + if (Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps) + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); diff --git a/htdocs/js/GraphTool/quadratictool.js b/htdocs/js/GraphTool/quadratictool.js index a12d62daa..b0fcde7ba 100644 --- a/htdocs/js/GraphTool/quadratictool.js +++ b/htdocs/js/GraphTool/quadratictool.js @@ -199,13 +199,12 @@ }; this.phase2 = (coords) => { - // Don't allow the second point to be created on the same - // vertical line as the first point or off the board. - if ( - this.point1.X() == gt.snapRound(coords[1], gt.snapSizeX) || - !gt.boardHasPoint(coords[1], coords[2]) - ) - return; + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same vertical line as the first point, + // then use the highlight point coordinates instead. + if (Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps) + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); @@ -231,14 +230,15 @@ }; this.phase3 = (coords) => { - // Don't allow the third point to be created on the same vertical line as the - // first point, on the same vertical line as the second point, or off the board. + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same vertical line as the first point, or on the same + // vertical line as the second point, then use the highlight point coordinates instead. if ( - this.point1.X() == gt.snapRound(coords[1], gt.snapSizeX) || - this.point2.X() == gt.snapRound(coords[1], gt.snapSizeX) || - !gt.boardHasPoint(coords[1], coords[2]) + Math.abs(this.point1.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps || + Math.abs(this.point2.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps ) - return; + coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); diff --git a/htdocs/js/GraphTool/sinewavetool.js b/htdocs/js/GraphTool/sinewavetool.js index 903113274..94e8ce576 100644 --- a/htdocs/js/GraphTool/sinewavetool.js +++ b/htdocs/js/GraphTool/sinewavetool.js @@ -225,8 +225,8 @@ this.supportsSolidDash = true; this.phase1 = (coords) => { - // If the current coordinates are off the board, then use the highlight point coordinates instead. - if (!gt.boardHasPoint(coords[1], coords[2])) coords = this.hlObjs.hl_point.coords.usrCoords; + // Don't allow the point to be created off the board + if (!gt.boardHasPoint(coords[1], coords[2])) return; gt.board.off('up'); @@ -252,12 +252,11 @@ }; this.phase2 = (coords) => { - // If the current coordinates are on the same vertical line as the first point or off the board, - // then use the coordinates of the highlight point instead. - if ( - !gt.boardHasPoint(coords[1], coords[2]) || - Math.abs(this.shiftPoint.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps - ) + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same vertical line as the first point, + // then use the highlight point coordinates instead. + if (Math.abs(this.shiftPoint.X() - gt.snapRound(coords[1], gt.snapSizeX)) < JXG.Math.eps) coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up'); @@ -291,12 +290,11 @@ }; this.phase3 = (coords) => { - // If the current coordinates are on the same horizontal line as the first point or off the board, + if (!gt.boardHasPoint(coords[1], coords[2])) return; + + // If the current coordinates are on the same horizontal line as the first point, // then use the highlight point coordinates instead. - if ( - Math.abs(this.shiftPoint.Y() - gt.snapRound(coords[2], gt.snapSizeY)) < JXG.Math.eps || - !gt.boardHasPoint(coords[1], coords[2]) - ) + if (Math.abs(this.shiftPoint.Y() - gt.snapRound(coords[2], gt.snapSizeY)) < JXG.Math.eps) coords = this.hlObjs.hl_point.coords.usrCoords; gt.board.off('up');