用path畫一個(gè)抽象的樹葉

源碼地址:https://github.com/X-FAN/LeafView
只是個(gè)簡(jiǎn)單的demo,大家可以參考下

public class PathTestView extends View {
    private int mWidth;
    private int mHeight;
    private int mDuration = 5000;
    private int mState = 0;//當(dāng)前狀態(tài);
    private float mFraction;


    private Paint mPaint;
    private PathMeasure mPathMeasure;
    private PathMeasure mPathMeasureLeft;
    private PathMeasure mPathMeasureRight;
    private ValueAnimator mAnimator;


    public PathTestView(Context context) {
        this(context, null);
    }

    public PathTestView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public PathTestView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initPaint();
        initPathAndMeasure();
        initAnimator();
    }

    private void initPaint() {
        mPaint = new Paint();
        mPaint.setStrokeWidth(10);
        mPaint.setStyle(Paint.Style.STROKE);
        mPaint.setColor(Color.WHITE);
    }

    private void initPathAndMeasure() {
        Path mPathLine = new Path();
        Path mPathLeft = new Path();
        Path mPathRight = new Path();
        mPathLine.lineTo(0, 400);
        mPathMeasure = new PathMeasure(mPathLine, false);
        mPathLeft.quadTo(-200, 200, 0, 400);//畫貝賽爾曲線
        mPathMeasureLeft = new PathMeasure(mPathLeft, false);
        mPathRight.quadTo(200, 200, 0, 400);
        mPathMeasureRight = new PathMeasure(mPathRight, false);
    }

    private void initAnimator() {
        mAnimator = ValueAnimator.ofFloat(0f, 1.0f);
        mAnimator.setDuration(mDuration);
        mAnimator.setInterpolator(new DecelerateInterpolator());
        mAnimator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
            @Override
            public void onAnimationUpdate(ValueAnimator animation) {
                mFraction = (float) animation.getAnimatedValue();
                invalidate();
            }
        });
        mAnimator.addListener(new AnimatorListenerAdapter() {
            @Override
            public void onAnimationEnd(Animator animation) {
                if (mState < 3) {//更改狀態(tài)
                    mState++;
                } else {
                    mState = 0;
                }
                mAnimator = mAnimator.clone();//本來想直接復(fù)用mAnimator,但是執(zhí)行到onAnimationEnd,mAnimator貌似沒有立即結(jié)束,直接start會(huì)有問題,
                mAnimator.start();           //問題待研究,若有知道具體原因的望告知

            }
        });
        mAnimator.start();
    }

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        mWidth = w;
        mHeight = h;
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        canvas.translate(mWidth / 2, mHeight / 2);// 將畫布坐標(biāo)原點(diǎn)移動(dòng)到中心位置
        canvas.scale(1, -1);//翻轉(zhuǎn)y軸
        canvas.save();
        drawLine(canvas);
        drawLeaf(canvas);
        canvas.restore();
    }

    /**
     * 畫葉子的邊界
     *
     * @param canvas
     */
    private void drawLeaf(Canvas canvas) {
        Path dst = new Path();
        Path dstRight = new Path();
        switch (mState) {
            case 0:
                mPathMeasureLeft.getSegment(0, mPathMeasureLeft.getLength() * mFraction, dst, true);//取出要繪制部分的path
                mPathMeasureRight.getSegment(0, mPathMeasureRight.getLength() * mFraction, dstRight, true);
                break;
            case 1:
                mPathMeasureLeft.getSegment(mPathMeasureLeft.getLength() * mFraction, mPathMeasureLeft.getLength(), dst, true);//取出要繪制部分的path
                mPathMeasureRight.getSegment(mPathMeasureRight.getLength() * mFraction, mPathMeasureRight.getLength(), dstRight, true);
                break;
            case 2:
                mPathMeasureLeft.getSegment(mPathMeasureLeft.getLength() * (1 - mFraction), mPathMeasureLeft.getLength(), dst, true);//取出要繪制部分的path
                mPathMeasureRight.getSegment(mPathMeasureRight.getLength() * (1 - mFraction), mPathMeasureRight.getLength(), dstRight, true);
                break;
            case 3:
                mPathMeasureLeft.getSegment(0, mPathMeasureLeft.getLength() * (1 - mFraction), dst, true);//取出要繪制部分的path
                mPathMeasureRight.getSegment(0, mPathMeasureRight.getLength() * (1 - mFraction), dstRight, true);
                break;
            default:
                break;
        }
        canvas.drawPath(dst, mPaint);
        canvas.drawPath(dstRight, mPaint);
    }

    /**
     * 畫主干
     */
    private void drawLine(Canvas canvas) {
        Path dst = new Path();
        switch (mState) {
            case 0:
                mPathMeasure.getSegment(0, mPathMeasure.getLength() * mFraction, dst, true);
                break;
            case 1:
                mPathMeasure.getSegment(mPathMeasure.getLength() * mFraction, mPathMeasure.getLength(), dst, true);
                break;
            case 2:
                mPathMeasure.getSegment(mPathMeasure.getLength() * (1 - mFraction), mPathMeasure.getLength(), dst, true);
                break;
            case 3:
                mPathMeasure.getSegment(0, mPathMeasure.getLength() * (1 - mFraction), dst, true);
                break;
            default:
                break;
        }
        canvas.drawPath(dst, mPaint);
    }


}

效果圖


這里寫圖片描述
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • 太長(zhǎng)了,還是轉(zhuǎn)載吧...今天在看博客的時(shí)候,無意中發(fā)現(xiàn)了@Trinea在GitHub上的一個(gè)項(xiàng)目Android開源...
    龐哈哈哈12138閱讀 20,274評(píng)論 3 283
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 173,314評(píng)論 25 708
  • 老住處的蚊子真是太熱情、太大方了! 昨日難得回去一趟,甫至樓前,闊別已久的蚊子撲面而來、與我相擁,互致問候,你叮我...
    土人行閱讀 191評(píng)論 0 0
  • 努力找不對(duì)方法,累死也是枉然!致自己!還好我在進(jìn)步,不要想那么多了,做好目前該做的,不要想太多!
    桐華韻錦閱讀 140評(píng)論 0 0
  • 今天看到一則新聞,內(nèi)容本身沒有什么,無非是某女當(dāng)年愛上某年長(zhǎng)二十歲男星,后來分手時(shí)泣不成聲云云,這樣的事不僅娛樂圈...
    洋小舟閱讀 233評(píng)論 0 2