How do I draw an arrowhead (in Android)?

后端 未结 8 477
一个人的身影
一个人的身影 2020-11-29 06:29

I\'m fairly new to Android and have been toying around with Canvas. I\'m attempting to draw an arrow but I\'m only having luck with drawing the shaft, none of the arrowhead

相关标签:
8条回答
  • 2020-11-29 07:10

    I try this code it has been working perfectly:

    switch (event.getAction())
    {
       case MotionEvent.ACTION_DOWN:
            mPath.reset();
            mPath.moveTo(x, y);
            mX = x;
            mY = y;
            startPoint = new PointF(event.getX(), event.getY());
            endPoint = new PointF();
            invalidate();
            break;
        case MotionEvent.ACTION_MOVE:
                float dx = Math.abs(x - mX);
            System.out.println("action move");
            float dy = Math.abs(y - mY);
            if (dx >= TOUCH_TOLERANCE || dy >= TOUCH_TOLERANCE)
            {
            //  currentDrawingPath.path.quadTo(mX,mY,(x + mX)/2, (y + mY)/2);
            }
            mX = x;
            mY = y;
              endPoint.x = event.getX();
              endPoint.y = event.getY();
              isDrawing = true;
              invalidate();
            break;
        case MotionEvent.ACTION_UP:
               mPath.lineTo(mX, mY);
               float deltaX =   endPoint.x-startPoint.x;
               float deltaY =   endPoint.y-startPoint.y;
               float frac = (float) 0.1;
         float point_x_1 = startPoint.x + (float) ((1 - frac) * deltaX + frac * deltaY);
         float point_y_1 = startPoint.y + (float) ((1 - frac) * deltaY - frac * deltaX);
               float point_x_2 = endPoint.x;
               float point_y_2 = endPoint.y;
         float point_x_3 = startPoint.x + (float) ((1 - frac) * deltaX - frac * deltaY);
         float point_y_3 = startPoint.y + (float) ((1 - frac) * deltaY + frac * deltaX);
               mPath.moveTo(point_x_1, point_y_1);
               mPath.lineTo(point_x_2, point_y_2);
               mPath.lineTo(point_x_3, point_y_3);
               mPath.lineTo(point_x_1, point_y_1);
               mPath.lineTo(point_x_1, point_y_1);
                mCanvas.drawPath(mPath, ppaint);
                endPoint.x = event.getX();
                endPoint.y = event.getY();
                isDrawing = false;
                invalidate();
            break;
        default:
            break;
    }       
    
    0 讨论(0)
  • 2020-11-29 07:11

    I've been having the same problem, I need an arrow to point in a certain direction. After playing around with drawing algorithms I decided the simplest method is to use a bitmap & simply use a Matrix to rotate it, e.g.

    ImageView image = (ImageView) findViewById(R.id.bitmap_image);
    Bitmap bMap = BitmapFactory.decodeResource(getResources(), R.drawable.test);
    Matrix mat = new Matrix();
    mat.postRotate(90);
    Bitmap bMapRotate = Bitmap.createBitmap(bMap, 0, 0, bMap.getWidth(), bMap.getHeight(), mat, true);
    image.setImageBitmap(bMapRotate);
    

    then your bitmap can be any fancy looking arrow you like.

    0 讨论(0)
  • 2020-11-29 07:15

    If you are looking for the solution to draw thousands of arrows under a second, with fixed length head lines, try this function (draws only arrow heads):

    private void fillArrow(Paint paint, Canvas canvas, float x0, float y0, float x1, float y1) {
        paint.setStyle(Paint.Style.STROKE);
    
        int arrowHeadLenght = 10;
        int arrowHeadAngle = 45;
        float[] linePts = new float[] {x1 - arrowHeadLenght, y1, x1, y1};
        float[] linePts2 = new float[] {x1, y1, x1, y1 + arrowHeadLenght};
        Matrix rotateMat = new Matrix();
    
        //get the center of the line
        float centerX = x1;
        float centerY = y1;
    
        //set the angle
        double angle = Math.atan2(y1 - y0, x1 - x0) * 180 / Math.PI + arrowHeadAngle;
    
        //rotate the matrix around the center
        rotateMat.setRotate((float) angle, centerX, centerY);
        rotateMat.mapPoints(linePts);
        rotateMat.mapPoints(linePts2);
    
        canvas.drawLine(linePts [0], linePts [1], linePts [2], linePts [3], paint);
        canvas.drawLine(linePts2 [0], linePts2 [1], linePts2 [2], linePts2 [3], paint);
    }
    

    Based on https://gamedev.stackexchange.com/questions/44456/drawing-lines-on-android-with-matrix

    0 讨论(0)
  • 2020-11-29 07:15

    Copypast from this answer https://stackoverflow.com/a/29383352/9975029

    private void fillArrow(Canvas canvas, float x0, float y0, float x1, float y1) {
        paint.setStyle(Paint.Style.FILL);
    
        float deltaX = x1 - x0;
        float deltaY = y1 - y0;
        double distance = Math.sqrt((deltaX * deltaX) + (deltaY * deltaY));
        float frac = (float) (1 / (distance / 30));
    
        float point_x_1 = x0 + (float) ((1 - frac) * deltaX + frac * deltaY);
        float point_y_1 = y0 + (float) ((1 - frac) * deltaY - frac * deltaX);
    
        float point_x_2 = x1;
        float point_y_2 = y1;
    
        float point_x_3 = x0 + (float) ((1 - frac) * deltaX - frac * deltaY);
        float point_y_3 = y0 + (float) ((1 - frac) * deltaY + frac * deltaX);
    
        Path path = new Path();
        path.setFillType(Path.FillType.EVEN_ODD);
    
        path.moveTo(point_x_1, point_y_1);
        path.lineTo(point_x_2, point_y_2);
        path.lineTo(point_x_3, point_y_3);
        path.lineTo(point_x_1, point_y_1);
        path.lineTo(point_x_1, point_y_1);
        path.close();
    
        canvas.drawPath(path, paint);
    }
    
    0 讨论(0)
  • 2020-11-29 07:19

    My Arrow Drawing code, maybe it can be of some use for somebody:

        /**
     * Draw an arrow
     * change internal radius and angle to change appearance
     * - angle : angle in degrees of the arrows legs
     * - radius : length of the arrows legs
     * @author Steven Roelants 2017
     *
     * @param paint
     * @param canvas
     * @param from_x
     * @param from_y
     * @param to_x
     * @param to_y
     */
    private void drawArrow(Paint paint, Canvas canvas, float from_x, float from_y, float to_x, float to_y)
    {
        float angle,anglerad, radius, lineangle;
    
        //values to change for other appearance *CHANGE THESE FOR OTHER SIZE ARROWHEADS*
        radius=10;
        angle=15;
    
        //some angle calculations
        anglerad= (float) (PI*angle/180.0f);
        lineangle= (float) (atan2(to_y-from_y,to_x-from_x));
    
        //tha line
        canvas.drawLine(from_x,from_y,to_x,to_y,paint);
    
        //tha triangle
        Path path = new Path();
        path.setFillType(Path.FillType.EVEN_ODD);
        path.moveTo(to_x, to_y);
        path.lineTo((float)(to_x-radius*cos(lineangle - (anglerad / 2.0))),
                (float)(to_y-radius*sin(lineangle - (anglerad / 2.0))));
        path.lineTo((float)(to_x-radius*cos(lineangle + (anglerad / 2.0))),
                (float)(to_y-radius*sin(lineangle + (anglerad / 2.0))));
        path.close();
    
        canvas.drawPath(path, paint);
    }
    
    0 讨论(0)
  • Here is code working perfect for me draw arrow head while drawing line on canvas

    package com.example.canvasexample;
    
    import android.content.Context;
    import android.graphics.Canvas;
    import android.graphics.Color;
    import android.graphics.Matrix;
    import android.graphics.Paint;
    import android.graphics.Path;
    import android.support.annotation.NonNull;
    import android.util.AttributeSet;
    import android.util.Log;
    import android.view.MotionEvent;
    import android.view.View;
    
    import java.util.ArrayList;
    
    import static android.view.MotionEvent.ACTION_DOWN;
    import static android.view.MotionEvent.ACTION_MOVE;
    import static android.view.MotionEvent.ACTION_UP;
    
    public class DrawerViewArrow extends View {
        private ArrayList<Path> drawingLinePath;
        private ArrayList<Path> drawingArrowPath;
        private ArrayList<Paint> drawingLinePaint;
        private int pathIndex = 0;
        private float startX = -1, startY = -1;
        private float mX = -1, mY = -1;
    
        public int arrowLength = 80;
        public int arrowWidth = 45;
        public int strokeWidth = 10;
    
        public DrawerViewArrow(Context context) {
            super(context);
            initPath();
        }
    
        public DrawerViewArrow(Context context, @NonNull AttributeSet attrs) {
            super(context, attrs);
            initPath();
        }
    
        public DrawerViewArrow(Context context, @NonNull AttributeSet attrs, int defStyleAttr) {
            super(context, attrs, defStyleAttr);
            initPath();
        }
    
        private Paint initPaint() {
            Paint mPaint = new Paint();
            mPaint.setAntiAlias(true);
            mPaint.setDither(true);
            mPaint.setColor(Color.GREEN);
            mPaint.setStyle(Paint.Style.FILL_AND_STROKE);
            mPaint.setStrokeJoin(Paint.Join.ROUND);
            mPaint.setStrokeCap(Paint.Cap.ROUND);
            mPaint.setStrokeWidth(strokeWidth);
            return mPaint;
        }
    
        private void initPath() {
            drawingLinePath = new ArrayList<>();
            drawingArrowPath = new ArrayList<>();
            drawingLinePath.add(new Path());
            drawingArrowPath.add(new Path());
            drawingLinePaint = new ArrayList<>();
            drawingLinePaint.add(initPaint());
            pathIndex++;
        }
    
        private Path createPath(MotionEvent event) {
            Path path = new Path();
            path.moveTo(event.getX(), event.getY());
            return path;
        }
    
        private void updateIndex(MotionEvent event) {
            if (pathIndex == drawingLinePath.size()) {
                drawingLinePath.add(createPath(event));
                drawingArrowPath.add(createPath(event));
                drawingLinePaint.add(initPaint());
                pathIndex++;
            }
        }
    
    
        @Override
        protected void onDraw(Canvas canvas) {
            super.onDraw(canvas);
    
            if (startX > -1 && mX > -1) {
                canvas.drawLine(startX, startY, mX, mY, initPaint());
                drawArrow(canvas);
            }
    
            for (int index = 0; index < pathIndex; index++) {
                Path path = drawingLinePath.get(index);
                Path arrow_path = drawingArrowPath.get(index);
                Paint paint = drawingLinePaint.get(index);
                canvas.drawPath(path, paint);
                canvas.drawPath(arrow_path, paint);
            }
        }
    
        private void drawArrow(Canvas canvas) {
            double angle = calculateAngle(startX, startY, mX, mY);
    
            float final_angle = (float) (180 - angle);
    
            Path arrow_path = new Path();
    
            Matrix arrow_matrix = new Matrix();
    
            arrow_matrix.postRotate(final_angle, mX, mY);
    
            arrow_path.moveTo(mX, mY);
            arrow_path.lineTo(mX - arrowWidth, mY + arrowLength);
            arrow_path.moveTo(mX, mY);
            arrow_path.lineTo(mX + arrowWidth, mY + arrowLength);
            arrow_path.lineTo(mX - (arrowWidth), mY + arrowLength);
            arrow_path.transform(arrow_matrix);
    
            canvas.drawPath(arrow_path, initPaint());
        }
    
        private void saveArrow() {
            if (mX == -1 || mY == -1) {
                return;
            }
    
            double angle = calculateAngle(startX, startY, mX, mY);
    
            float final_angle = (float) (180 - angle);
    
            Path arrow_path = drawingArrowPath.get(pathIndex - 1);
    
            Matrix arrow_matrix = new Matrix();
    
            arrow_matrix.postRotate(final_angle, mX, mY);
    
            arrow_path.moveTo(mX, mY);
            arrow_path.lineTo(mX - arrowWidth, mY + arrowLength);
            arrow_path.moveTo(mX, mY);
            arrow_path.lineTo(mX + arrowWidth, mY + arrowLength);
            arrow_path.lineTo(mX - (arrowWidth), mY + arrowLength);
            arrow_path.transform(arrow_matrix);
        }
    
        public double calculateAngle(double x1, double y1, double x2, double y2) {
            double angle = Math.toDegrees(Math.atan2(x2 - x1, y2 - y1));
    
            angle = angle + Math.ceil(-angle / 360) * 360; //Keep angle between 0 and 360
    
            return angle;
        }
    
        @Override
        public boolean onTouchEvent(MotionEvent event) {
            switch (event.getAction()) {
                case ACTION_UP:
                    actionUp(event);
                    break;
                case ACTION_MOVE:
                    actionMove(event);
                    break;
                case ACTION_DOWN:
                    actionDown(event);
                    break;
            }
            invalidate();
            return true;
        }
    
        private void actionDown(MotionEvent event) {
            updateIndex(event);
            startX = event.getX();
            startY = event.getY();
        }
    
        private void actionMove(MotionEvent event) {
            mX = event.getX();
            mY = event.getY();
        }
    
        private void actionUp(MotionEvent event) {
            drawingLinePath.get(pathIndex - 1).lineTo(event.getX(), event.getY());
            saveArrow();
            startX = -1;
            startY = -1;
            mX = -1;
            mY = -1;
        }
    }
    
    0 讨论(0)
提交回复
热议问题