Как анимировать путь на холсте - андроид
Можно ли подключить аниматор к пути? Есть ли другой способ рисовать на анимированных линиях холста? Я искал это до публикации, но об этом ничего нет. В двух других сообщениях здесь и здесь ходят решения и не подходят мне.
Спасибо!
Я отправляю свой код внутри метода onDraw, чтобы указать, что именно я хочу.
paint.setStyle(Paint.Style.STROKE);
paint.setStrokeWidth(2);
paint.setColor(Color.BLACK);
Path path = new Path();
path.moveTo(10, 50); // THIS TRANSFORMATIONS TO BE ANIMATED!!!!!!!!
path.lineTo(40, 50);
path.moveTo(40, 50);
path.lineTo(50, 40);
// and so on...
canvas.drawPath(path, paint);
Любые идеи????
Ответы
Ответ 1
Вы можете преобразовать свой холст по времени, то есть:
class MyView extends View {
int framesPerSecond = 60;
long animationDuration = 10000; // 10 seconds
Matrix matrix = new Matrix(); // transformation matrix
Path path = new Path(); // your path
Paint paint = new Paint(); // your paint
long startTime;
public MyView(Context context) {
super(context);
// start the animation:
this.startTime = System.currentTimeMillis();
this.postInvalidate();
}
@Override
protected void onDraw(Canvas canvas) {
long elapsedTime = System.currentTimeMillis() - startTime;
matrix.postRotate(30 * elapsedTime/1000); // rotate 30° every second
matrix.postTranslate(100 * elapsedTime/1000, 0); // move 100 pixels to the right
// other transformations...
canvas.concat(matrix); // call this before drawing on the canvas!!
canvas.drawPath(path, paint); // draw on canvas
if(elapsedTime < animationDuration)
this.postInvalidateDelayed( 1000 / framesPerSecond);
}
}
Ответ 2
попробуйте следующее:
class PathDrawable extends Drawable implements AnimatorUpdateListener {
private Path mPath;
private Paint mPaint;
private ValueAnimator mAnimator;
public PathDrawable() {
mPath = new Path();
mPaint = new Paint();
mPaint.setColor(0xffffffff);
mPaint.setStrokeWidth(5);
mPaint.setStyle(Style.STROKE);
}
public void startAnimating() {
Rect b = getBounds();
mAnimator = ValueAnimator.ofInt(-b.bottom, b.bottom);
mAnimator.setDuration(1000);
mAnimator.addUpdateListener(this);
mAnimator.start();
}
@Override
public void draw(Canvas canvas) {
canvas.drawPath(mPath, mPaint);
}
@Override
public void setAlpha(int alpha) {
}
@Override
public void setColorFilter(ColorFilter cf) {
}
@Override
public int getOpacity() {
return PixelFormat.TRANSLUCENT;
}
@Override
public void onAnimationUpdate(ValueAnimator animator) {
mPath.reset();
Rect b = getBounds();
mPath.moveTo(b.left, b.bottom);
mPath.quadTo((b.right-b.left)/2, (Integer) animator.getAnimatedValue(), b.right, b.bottom);
invalidateSelf();
}
}
чтобы проверить его, добавьте в свой метод onCreate:
TextView view = new TextView(this);
view.setText("click me");
view.setTextColor(0xffcccccc);
view.setGravity(Gravity.CENTER);
view.setTextSize(48);
final PathDrawable d = new PathDrawable();
view.setBackgroundDrawable(d);
OnClickListener l = new OnClickListener() {
@Override
public void onClick(View v) {
d.startAnimating();
}
};
view.setOnClickListener(l);
setContentView(view);
Ответ 3
Вы можете создать PathMeasure
для своего пути и определить его длину:
private PathMeasure pathMeasure; // field
// After you've created your path
pathMeasure = new PathMeasure(path, false);
pathLength = pathMeasure.getLength();
Затем вы можете получить конкретную точку пути, используя getPosTan() внутри, скажем, прослушиватель обновлений ValueAnimator:
final float[] position = new float[2]; // field
// Inside your animation update method, where dt is your 0..1 animated fraction
final float distance = dt * pathLength;
pathMeasure.getPosTan(distance, position, null);
// If using onDraw you'll need to tell the view to redraw using the new position
invalidate();
Затем вы можете использовать позицию в onDraw (или что-то еще).
canvas.drawCircle(position[0], position[1], radius, paint);
Преимущества такого подхода заключаются в том, что он прост, не требует коротких математических вычислений и работает на всех API.
Если вы используете API 21+, вы можете использовать ValueAnimator и передать путь, чтобы использовать его позиции, что проще. Пример SO вопрос.