preface
To keep track of what’s drawn, many applications, such as drawing apps, computer-Aided Design System (CAD) systems, and games, maintain a list of the objects currently displayed. In general, these applications allow the user to manipulate objects currently displayed on the screen. For example, in CAD applications, we can select, move, scale and so on elements in the design.
The same is true in Canvas, which provides an API called isPointInPath(x, y) to determine whether the point (x, y) is in the path. Returns true if it is in a path. So we can think of the following:
Maintain an array that can describe each path, judge whether the click position is in a certain path through ispointInPath(x, y), select this path, perform operations (move, zoom, etc.), and then draw a graph
There are previous examples of the same type: This article records how to use Canvas to drag and edit a Bezier curve. The example comes from HTML5 Canvas Core Technology, and some codes have been modified. The Demo is also put in CodePen as follows (the printing is for screen recording software 👹) :
CodePen open
- Thinking on
- “Rubber band” and Bezier curves
- Implementation logic
- Pseudocode thinking
- The code segment
- mousedown
- mousemove
- mouseup
- conclusion
- The resources
Thinking on
The idea award is mainly divided into three parts, namely “Rubber band” and Bezier curve, implementation logic, pseudo-code idea “Rubber Band” and Bezier curve: explain how to draw the Demo bezier curve implementation logic: This Demo has both drawing and editing two state pseudo-code ideas: combined with pseudo-code to elaborate ideas
“Rubber band” and Bezier curves
APIvoid CTX. BezierCurveTo (cp1x, CP1y, CP2x, cp2y, x, y); Cp1 and cp2 represent two control points, with x and y representing the position of the other endpoint and moveTo() as the starting position of the other endpoint
How do I draw it next
rubberBandRect
Left, top, width, height
Curves are drawn or edited by changing rubberBandRect and then drawing corresponding points and curves
Implementation logic
This stage is the edit state and occurs after the curve is drawn, i.e., it is somewhat draggable:
This state is a drawable state, after the initial state or the end of the editing state (this Demo does not edit the “defined curve” function) :
To sum up the above and Demo experience, we can summarize the logic as follows:
Pseudocode thinking
This part is explained in combination with some key variables and mind maps in the code: key variables:
Editing: // drawingCurve: // Whether a curve is being drawn draggingPoint: // Record drag point, possible values include //null(click blank part after drawing curve, no drag point, end editing phase), //endPoint,controlPointCopy the code
The code segment
mousedown
canvas.onmousedown = function(e) { const loc = windowToCanvas(e.clientX, e.clientY); // Get the position e.preventDefault();if (editing === false) {// Draw a line saveDrawingSurface(); mousedown.x = loc.x; mousedown.y = loc.y; updateRubberBandRectangle(loc); // Update rubberBandRect drawingCurve =true; // Mark is drawing}else{// Edit curve stage enter // select click position corresponding point, three possible null, endPoint, controlPoint draggingPoint = cursorInControlPoint(LOC);if(draggingPoint === null) { draggingPoint = cursorInEndPoint(loc); }}}Copy the code
Mousedown determines whether to draw or edit a curve
mousemove
canvas.onmousemove = function (e) {
const loc = windowToCanvas(e.clientX, e.clientY);
if(drawingCurve | | draggingPoint) {/ / mouse movement interference without using application e.p reventDefault (); restoreDrawingSurface();if(guideWires) { drawGuideWires(loc.x, loc.y); }}if(drawingCurve) {// Draw the curve updateRubberBand(loc); // Update rubberBand and draw drawControlAndEndPoints(); }else if(draggingPoint) {// MouseDown starts dragging updateDraggingPoint(LOC) by clicking on an endpoint or control point; drawControlAndEndPoints(); drawBezierCurve(); }}Copy the code
Mousemove is designed to eliminate mouse-over and phase drawing when not using the mousemove function
mouseup
canvas.onmouseup = function (e) {
const loc = windowToCanvas(e.clientX, e.clientY);
restoreDrawingSurface();
if (editing === false) {//画线阶段进入
updateRubberBand(loc);
drawControlAndEndPoints();
drawingCurve = false;
editing = true; // After mouseup, you can edit the curve.else{// Edit curve stage entersif(draggingPoint) {// drawControlAndEndPoints(); }else{// No drop point, end editing =false; } drawBezierCurve(); draggingPoint = null; }}Copy the code
Mouseup is mainly the subsequent processing of each stage. If the drawing stage is finished, the editing stage will be started; if the editing stage is finished, the editing stage will be continued or finished by judging whether there is a drag point
conclusion
This example will be much easier once RubberBandRect and the controls for the drawing and editing phases are clarified
The resources
HTML5 Canvas Core Technology