Maison interface Web tutoriel HTML animation之objectAnimator_html/css_WEB-ITnose

animation之objectAnimator_html/css_WEB-ITnose

Jun 24, 2016 am 11:40 AM

这次借助github上的开源项目ShapeLoadingView来学习下ObjectAnimator和animatorSet.
代码结构目录:

  1. ShapeLoadingView.java
  2. LoadingView.java
    LoadingView是绘制三个基本图形的类。
    ShapeLoadingView初始化图形并操作图形进行动画。
    下面上加了注释的代码:
package com.mingle.widget;import android.annotation.TargetApi;import android.content.Context;import android.content.res.TypedArray;import android.os.Build;import android.text.TextUtils;import android.util.AttributeSet;import android.view.Gravity;import android.view.LayoutInflater;import android.view.View;import android.view.ViewGroup;import android.view.animation.AccelerateInterpolator;import android.view.animation.DecelerateInterpolator;import android.widget.FrameLayout;import android.widget.ImageView;import android.widget.TextView;import com.mingle.shapeloading.R;import com.nineoldandroids.animation.Animator;import com.nineoldandroids.animation.AnimatorSet;import com.nineoldandroids.animation.ObjectAnimator;/** * Created by zzz40500 on 15/4/6. */public class LoadingView extends FrameLayout {    private static final int ANIMATION_DURATION = 500;    private static  float mDistance = 200;    private ShapeLoadingView mShapeLoadingView;    private ImageView mIndicationIm;    private TextView mLoadTextView;    private int mTextAppearance;    private String mLoadText;    public LoadingView(Context context) {        super(context);    }    public LoadingView(Context context, AttributeSet attrs) {        //构造函数        super(context, attrs, 0);        init(context, attrs);    }    private void init(Context context, AttributeSet attrs) {        //这里是通过自定义属性来显示字符串        TypedArray typedArray = context                .obtainStyledAttributes(attrs, R.styleable.LoadingView);        mLoadText = typedArray.getString(R.styleable.LoadingView_loadingText);        mTextAppearance = typedArray.getResourceId(R.styleable.LoadingView_loadingTextAppearance, -1);        typedArray.recycle();    }    public LoadingView(Context context, AttributeSet attrs, int defStyleAttr) {        //构造函数        super(context, attrs, defStyleAttr);        init(context, attrs);    }    //这里定义了一个针对LL版本的构造函数,我这可能因为sdk版本这里会报错,如果报错注释掉就行了    @TargetApi(Build.VERSION_CODES.LOLLIPOP)    public LoadingView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {        super(context, attrs, defStyleAttr, defStyleRes);        init(context, attrs);    }    //dp和像素的转换    public int dip2px(float dipValue) {        final float scale = getContext().getResources().getDisplayMetrics().density;        return (int) (dipValue * scale + 0.5f);    }    //引入布局    @Override    protected void onFinishInflate() {        super.onFinishInflate();        View view = LayoutInflater.from(getContext()).inflate(R.layout.load_view, null);        mDistance = dip2px(54f);        LayoutParams layoutParams = new LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT);        layoutParams.gravity = Gravity.CENTER;        mShapeLoadingView = (ShapeLoadingView) view.findViewById(R.id.shapeLoadingView);        mIndicationIm = (ImageView) view.findViewById(R.id.indication);        mLoadTextView = (TextView) view.findViewById(R.id.promptTV);        if (mTextAppearance != -1) {            mLoadTextView.setTextAppearance(getContext(), mTextAppearance);        }        setLoadingText(mLoadText);        //显示绘画布局        addView(view, layoutParams);        //这里是设计一个延时 每隔900调用一次跌落,相当于900ms是一次动画的周期        this.postDelayed(new Runnable() {            @Override            public void run() {                freeFall();            }        }, 900);    }    public void setLoadingText(CharSequence loadingText) {        if (TextUtils.isEmpty(loadingText)) {            mLoadTextView.setVisibility(GONE);        } else {            mLoadTextView.setVisibility(VISIBLE);        }        mLoadTextView.setText(loadingText);    }    /** * 上抛,上抛是动画的核心,上抛是两个组合动作:1,图形进行旋转;2,图形向上平移,同时还有下面阴影部分随着图形位置变化 * 进行的跟随变化。这里使用了ObjectAnimator来控制每个动画的动作,最后使用AnimatorSet将三个部分组合在一起。 * 看一下具体的动作 */    public void upThrow() {        //mShapeLoadingView就是LoadingView里面绘制的图形买第一个objectAnimator控制它进行平移        //使用objectAnimator.ofFloat及参数translationY来进行纵向的平移        ObjectAnimator objectAnimator = ObjectAnimator.ofFloat(mShapeLoadingView, "translationY", mDistance, 0);        //动画下部的阴影这里使用ofFloat及参数scaleX来进行X轴的缩放,02f-1是缩放比例 阴影在20%到100%之间变化        ObjectAnimator scaleIndication = ObjectAnimator.ofFloat(mIndicationIm, "scaleX", 0.2f, 1);        //这段是对图形做一个旋转的动作        ObjectAnimator objectAnimator1 = null;        switch (mShapeLoadingView.getShape()) {            case SHAPE_RECT:                objectAnimator1 = ObjectAnimator.ofFloat(mShapeLoadingView, "rotation", 0, -120);                break;            case SHAPE_CIRCLE:                objectAnimator1 = ObjectAnimator.ofFloat(mShapeLoadingView, "rotation", 0, 180);                break;            case SHAPE_TRIANGLE:                objectAnimator1 = ObjectAnimator.ofFloat(mShapeLoadingView, "rotation", 0, 180);                break;        }        //设置animation的持续时间,通过setDuration.        objectAnimator.setDuration(ANIMATION_DURATION);        objectAnimator1.setDuration(ANIMATION_DURATION);        //设置一个减速插值器        objectAnimator.setInterpolator(new DecelerateInterpolator(factor));        objectAnimator1.setInterpolator(new DecelerateInterpolator(factor));        AnimatorSet animatorSet = new AnimatorSet();        animatorSet.setDuration(ANIMATION_DURATION);        //animatorSet的方法playtogther让三个动画同时运行        animatorSet.playTogether(objectAnimator, objectAnimator1, scaleIndication);        animatorSet.addListener(new Animator.AnimatorListener() {            @Override            public void onAnimationStart(Animator animation) {            }            @Override            public void onAnimationEnd(Animator animation) {                freeFall();            }            @Override            public void onAnimationCancel(Animator animation) {            }            @Override            public void onAnimationRepeat(Animator animation) {            }        });        animatorSet.start();    }    public float factor = 1.2f;    /** * 下落 */    public void freeFall() {        //主要的点和上抛一致不讲了        ObjectAnimator objectAnimator = ObjectAnimator.ofFloat(mShapeLoadingView, "translationY", 0, mDistance);        ObjectAnimator scaleIndication = ObjectAnimator.ofFloat(mIndicationIm, "scaleX", 1, 0.2f);        objectAnimator.setDuration(ANIMATION_DURATION);        objectAnimator.setInterpolator(new AccelerateInterpolator(factor));        AnimatorSet animatorSet = new AnimatorSet();        animatorSet.setDuration(ANIMATION_DURATION);        animatorSet.playTogether(objectAnimator, scaleIndication);        animatorSet.addListener(new Animator.AnimatorListener() {            @Override            public void onAnimationStart(Animator animation) {            }            @Override            public void onAnimationEnd(Animator animation) {                //下落到底端改变图形                mShapeLoadingView.changeShape();                upThrow();            }            @Override            public void onAnimationCancel(Animator animation) {            }            @Override            public void onAnimationRepeat(Animator animation) {            }        });        animatorSet.start();    }}
Copier après la connexion
package com.mingle.widget;import android.annotation.TargetApi;import android.content.Context;import android.graphics.Canvas;import android.graphics.Paint;import android.graphics.Path;import android.os.Build;import android.util.AttributeSet;import android.view.View;import com.mingle.shapeloading.R;/** * Created by zzz40500 on 15/4/4. */public class ShapeLoadingView extends View {    private static final float genhao3 = 1.7320508075689f;    private static  final  float mTriangle2Circle =0.25555555f;    private Shape mShape = Shape.SHAPE_CIRCLE;    /** * 用贝赛尔曲线画圆 */    private float mMagicNumber = 0.55228475f;    public ShapeLoadingView(Context context) {        super(context);        init();    }    public ShapeLoadingView(Context context, AttributeSet attrs) {        super(context, attrs);        init();    }    public ShapeLoadingView(Context context, AttributeSet attrs, int defStyleAttr) {        super(context, attrs, defStyleAttr);    }    @TargetApi(Build.VERSION_CODES.LOLLIPOP)    public ShapeLoadingView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {        super(context, attrs, defStyleAttr, defStyleRes);        init();    }    private void init() {        mPaint = new Paint();        mPaint.setColor(getResources().getColor(R.color.triangle));        mPaint.setAntiAlias(true);        //看到网上说这个FILL_AND_STROKE有去锯齿的作用        mPaint.setStyle(Paint.Style.FILL_AND_STROKE);        setBackgroundColor(getResources().getColor(R.color.view_bg));    }    public boolean mIsLoading = false;    private Paint mPaint;    private float mControlX = 0;    private float mControlY = 0;    private float mAnimPercent;    @Override    protected void onDraw(Canvas canvas) {        super.onDraw(canvas);        //绘制三个图形的三角形方框圆形的位置,作者在这里标记动画可以优化,估计会有后续修改        if(getVisibility()==GONE){            return;        }        // FIXME: 15/6/15 动画待优化        switch (mShape) {            case SHAPE_TRIANGLE:                if (mIsLoading) {                    mAnimPercent += 0.1611113;                    // triangle to circle                    Path path = new Path();                    path.moveTo(relativeXFromView(0.5f), relativeYFromView(0f));                    if (mAnimPercent >= 1) {                        mShape = Shape.SHAPE_CIRCLE;                        mIsLoading = false;                        mAnimPercent=1;                    }                    float controlX = mControlX - relativeXFromView(mAnimPercent* mTriangle2Circle)                            * genhao3;                    float controlY = mControlY - relativeYFromView(mAnimPercent* mTriangle2Circle);                    path.quadTo(relativeXFromView(1) - controlX, controlY, relativeXFromView(0.5f + genhao3 / 4), relativeYFromView(0.75f));                    path.quadTo(relativeXFromView(0.5f), relativeYFromView(0.75f + 2 * mAnimPercent* mTriangle2Circle), relativeXFromView(0.5f - genhao3 / 4), relativeYFromView(0.75f));                    path.quadTo(controlX, controlY, relativeXFromView(0.5f), relativeYFromView(0f));                    path.close();                    canvas.drawPath(path, mPaint);                    invalidate();                } else {                    Path path = new Path();                    mPaint.setColor(getResources().getColor(R.color.triangle));                    path.moveTo(relativeXFromView(0.5f), relativeYFromView(0f));                    path.lineTo(relativeXFromView(1), relativeYFromView(genhao3 / 2f));                    path.lineTo(relativeXFromView(0), relativeYFromView(genhao3/2f));                    mControlX = relativeXFromView(0.5f - genhao3 / 8.0f);                    mControlY = relativeYFromView(3 / 8.0f);                    mAnimPercent = 0;                    path.close();                    canvas.drawPath(path, mPaint);                }                break;            case SHAPE_CIRCLE:                if (mIsLoading) {                    float magicNumber = mMagicNumber + mAnimPercent;                    mAnimPercent += 0.12;                    if (magicNumber + mAnimPercent >= 1.9f) {                        mShape = Shape.SHAPE_RECT;                        mIsLoading = false;                    }                    Path path = new Path();                    path.moveTo(relativeXFromView(0.5f), relativeYFromView(0f));                    path.cubicTo(relativeXFromView(0.5f + magicNumber / 2), relativeYFromView(0f),                            relativeXFromView(1), relativeYFromView(0.5f - magicNumber / 2),                            relativeXFromView(1f), relativeYFromView(0.5f));                    path.cubicTo(                            relativeXFromView(1), relativeXFromView(0.5f + magicNumber / 2),                            relativeXFromView(0.5f + magicNumber / 2), relativeYFromView(1f),                            relativeXFromView(0.5f), relativeYFromView(1f));                    path.cubicTo(relativeXFromView(0.5f - magicNumber / 2), relativeXFromView(1f),                            relativeXFromView(0), relativeYFromView(0.5f + magicNumber / 2),                            relativeXFromView(0f), relativeYFromView(0.5f));                    path.cubicTo(relativeXFromView(0f), relativeXFromView(0.5f - magicNumber / 2),                            relativeXFromView(0.5f - magicNumber / 2), relativeYFromView(0),                            relativeXFromView(0.5f), relativeYFromView(0f));                    path.close();                    canvas.drawPath(path, mPaint);                    invalidate();                } else {      mPaint.setColor(getResources().getColor(R.color.circle));                    Path path = new Path();                    float magicNumber = mMagicNumber;                    path.moveTo(relativeXFromView(0.5f), relativeYFromView(0f));                    path.cubicTo(relativeXFromView(0.5f + magicNumber / 2), 0,                            relativeXFromView(1), relativeYFromView(magicNumber / 2),                            relativeXFromView(1f), relativeYFromView(0.5f));                    path.cubicTo(                            relativeXFromView(1), relativeXFromView(0.5f + magicNumber / 2),                            relativeXFromView(0.5f + magicNumber / 2), relativeYFromView(1f),                            relativeXFromView(0.5f), relativeYFromView(1f));                    path.cubicTo(relativeXFromView(0.5f - magicNumber / 2), relativeXFromView(1f),                            relativeXFromView(0), relativeYFromView(0.5f + magicNumber / 2),                            relativeXFromView(0f), relativeYFromView(0.5f));                    path.cubicTo(relativeXFromView(0f), relativeXFromView(0.5f - magicNumber / 2),                            relativeXFromView(0.5f - magicNumber / 2), relativeYFromView(0),                            relativeXFromView(0.5f), relativeYFromView(0f));                    mAnimPercent = 0;                    path.close();                    canvas.drawPath(path, mPaint);                }                break;            case SHAPE_RECT:                if (mIsLoading) {                    mAnimPercent += 0.15;                    if (mAnimPercent >= 1) {                        mShape = Shape.SHAPE_TRIANGLE;                        mIsLoading = false;                        mAnimPercent = 1;                    }                    Path path = new Path();                    path.moveTo(relativeXFromView(0.5f * mAnimPercent), 0);                    path.lineTo(relativeYFromView(1 - 0.5f * mAnimPercent), 0);                    float distanceX = (mControlX) * mAnimPercent;                    float distanceY = (relativeYFromView(1f) - mControlY) * mAnimPercent;                    path.lineTo(relativeXFromView(1f) - distanceX, relativeYFromView(1f) - distanceY);                    path.lineTo(relativeXFromView(0f) + distanceX, relativeYFromView(1f) - distanceY);                    path.close();                    canvas.drawPath(path, mPaint);                    invalidate();                } else {                    mPaint.setColor(getResources().getColor(R.color.rect));                    mControlX = relativeXFromView(0.5f - genhao3 / 4);                    mControlY = relativeYFromView(0.75f);                    Path path = new Path();                    path.moveTo(relativeXFromView(0f), relativeYFromView(0f));                    path.lineTo(relativeXFromView(1f), relativeYFromView(0f));                    path.lineTo(relativeXFromView(1f), relativeYFromView(1f));                    path.lineTo(relativeXFromView(0f), relativeYFromView(1f));                    path.close();                    mAnimPercent = 0;                    canvas.drawPath(path, mPaint);                }                break;        }    }    private float relativeXFromView(float percent) {        return getWidth() * percent;    }    private float relativeYFromView(float percent) {        return getHeight() * percent;    }    public void changeShape() {        mIsLoading = true;        invalidate();    public enum Shape {        SHAPE_TRIANGLE, SHAPE_RECT, SHAPE_CIRCLE    }    @Override    public void setVisibility(int visibility) {        super.setVisibility(visibility);        if(visibility==VISIBLE){            invalidate();        }    }    public Shape getShape() {        return mShape;    }}
Copier après la connexion

这个开源项目我们学习的两个主要知识
1.使用path绘制图形
2.ObjectAnimation&AnimatorSet
看了这个项目是不是可以用这两个知识点做一个自己喜欢的动画?
just do it.

版权声明:本文为博主原创文章,未经博主允许不得转载。

Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

Video Face Swap

Video Face Swap

Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Outils chauds

Bloc-notes++7.3.1

Bloc-notes++7.3.1

Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise

SublimeText3 version chinoise

Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1

Envoyer Studio 13.0.1

Puissant environnement de développement intégré PHP

Dreamweaver CS6

Dreamweaver CS6

Outils de développement Web visuel

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Sujets chauds

Tutoriel Java
1663
14
Tutoriel PHP
1263
29
Tutoriel C#
1237
24
Comprendre HTML, CSS et JavaScript: un guide pour débutant Comprendre HTML, CSS et JavaScript: un guide pour débutant Apr 12, 2025 am 12:02 AM

WebDevelopmentReliesOnHTML, CSS, etjavascript: 1) HTMLSTRUCTURESCONTENT, 2) CSSSTYLESIT, et3) JavascriptAdddsInterActivity, Forming TheasisofmodernweBEBExperiences.

Les rôles de HTML, CSS et JavaScript: responsabilités de base Les rôles de HTML, CSS et JavaScript: responsabilités de base Apr 08, 2025 pm 07:05 PM

HTML définit la structure Web, CSS est responsable du style et de la mise en page, et JavaScript donne une interaction dynamique. Les trois exercent leurs fonctions dans le développement Web et construisent conjointement un site Web coloré.

HTML, CSS et JavaScript: outils essentiels pour les développeurs Web HTML, CSS et JavaScript: outils essentiels pour les développeurs Web Apr 09, 2025 am 12:12 AM

HTML, CSS et JavaScript sont les trois piliers du développement Web. 1. HTML définit la structure de la page Web et utilise des balises telles que, etc. 2. CSS contrôle le style de page Web, en utilisant des sélecteurs et des attributs tels que la couleur, la taille de la police, etc. 3. JavaScript réalise les effets dynamiques et l'interaction, par la surveillance des événements et les opérations DOM.

HTML: The Structure, CSS: The Style, Javascript: Le comportement HTML: The Structure, CSS: The Style, Javascript: Le comportement Apr 18, 2025 am 12:09 AM

Les rôles de HTML, CSS et JavaScript dans le développement Web sont: 1. HTML définit la structure de la page Web, 2. CSS contrôle le style de page Web, et 3. JavaScript ajoute un comportement dynamique. Ensemble, ils construisent le cadre, l'esthétique et l'interactivité des sites Web modernes.

L'avenir de HTML: évolution et tendances de la conception Web L'avenir de HTML: évolution et tendances de la conception Web Apr 17, 2025 am 12:12 AM

L'avenir de HTML est plein de possibilités infinies. 1) Les nouvelles fonctionnalités et normes comprendront plus de balises sémantiques et la popularité des composants Web. 2) La tendance de la conception Web continuera de se développer vers une conception réactive et accessible. 3) L'optimisation des performances améliorera l'expérience utilisateur grâce à des technologies de chargement d'image réactives et de chargement paresseux.

L'avenir de HTML, CSS et JavaScript: Tendances de développement Web L'avenir de HTML, CSS et JavaScript: Tendances de développement Web Apr 19, 2025 am 12:02 AM

Les tendances futures de HTML sont la sémantique et les composants Web, les tendances futures de CSS sont CSS-in-JS et CSShoudini, et les tendances futures de JavaScript sont WebAssembly et sans serveur. 1. La sémantique HTML améliore l'accessibilité et les effets de référencement, et les composants Web améliorent l'efficacité du développement, mais l'attention doit être accordée à la compatibilité du navigateur. 2. CSS-in-JS améliore la flexibilité de gestion du style mais peut augmenter la taille du fichier. CSShoudini permet le fonctionnement direct du rendu CSS. 3.WeBassembly optimise les performances de l'application du navigateur mais a une courbe d'apprentissage abrupte, et sans serveur simplifie le développement mais nécessite une optimisation des problèmes de démarrage à froid.

HTML vs CSS vs JavaScript: un aperçu comparatif HTML vs CSS vs JavaScript: un aperçu comparatif Apr 16, 2025 am 12:04 AM

Les rôles de HTML, CSS et JavaScript dans le développement Web sont: HTML est responsable de la structure du contenu, CSS est responsable du style et JavaScript est responsable du comportement dynamique. 1. HTML définit la structure et le contenu de la page Web via des balises pour assurer la sémantique. 2. CSS contrôle le style de page Web via des sélecteurs et des attributs pour le rendre beau et facile à lire. 3. JavaScript contrôle le comportement de la page Web via les scripts pour atteindre des fonctions dynamiques et interactives.

HTML: construire la structure des pages Web HTML: construire la structure des pages Web Apr 14, 2025 am 12:14 AM

HTML est la pierre angulaire de la construction de la structure des pages Web. 1. HTML définit la structure et la sémantique du contenu et les utilisations, etc. Tags. 2. Fournir des marqueurs sémantiques, tels que, etc., pour améliorer l'effet SEO. 3. Pour réaliser l'interaction de l'utilisateur via des balises, faites attention à la vérification de la forme. 4. Utilisez des éléments avancés tels que, combinés avec JavaScript pour obtenir des effets dynamiques. 5. Les erreurs courantes incluent des étiquettes non clôturées et des valeurs d'attribut non déposées et des outils de vérification sont nécessaires. 6. Les stratégies d'optimisation comprennent la réduction des demandes HTTP, la compression du HTML, l'utilisation de balises sémantiques, etc.

See all articles