仿爱奇艺加载dialog

本文详细介绍了如何在Android应用中创建类似爱奇艺视频加载时的动画效果。通过计算三角形的位置和使用旋转动画实现动态效果,文章提供了完整的代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

仿爱奇艺在加载视频时的动画,想上张图;



gif的效果有点卡,先说一下实现的方法,

主要的难点在于怎样将三角形画到空间的中间,我通过三角形中心到顶角的距离来确定三角形的大小,


当三角形在中间时角a的大小为60度,线段A的长度为已知的自己设定的,我这里设定的为40像素,则C的长度为

A乘以sin(a)  B的长度为A乘以cos(a);   这里假设空间的长宽各位200的正方形,所以三角形的上顶点的坐标为(100,60),左下顶点为(200 / 2 - A * cos(a)),右顶点的坐标为(200 / 2,A* sin(a)),确定了三角形的三个顶点就可以在通过drawPath画三角形了,代码如下:

path.moveTo(width / 2, height / 2 - length);
path.lineTo(formantNumber(width / 2 - Math.sin(Math.PI / 3) * length), width / 2 + length / 2);
path.lineTo(formantNumber(width / 2 + Math.sin(Math.PI / 3) * length), width / 2 + length / 2);
path.close();

当布局加载完成后开始动画,由于但旋转的时候弧形的扫过的角度为360度,旋转时看不出来圆弧的旋转,

/**
 * 布局加载完成后开始动画
 */
@Override
protected void onFinishInflate() {
    super.onFinishInflate();
    startAnimation(rotateAnimation);
}

/**
 * 初始化动画
 */
private void initRouteAnimation() {
    rotateAnimation = new RotateAnimation(0, 1800, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
    rotateAnimation.setInterpolator(new AccelerateDecelerateInterpolator());
    rotateAnimation.setDuration(1500);
    rotateAnimation.setFillAfter(false);
    rotateAnimation.setAnimationListener(this);
}

当动画执行完后将将扫过的角度设置为0度,重新绘制

/**
 * 动画介绍后的回调
 *
 * @param animation
 */
@Override
public void onAnimationEnd(Animation animation) {
    sweepAngle = 0;
    isAfterAnimation = true;
    invalidate();
}

下面是onDraw方法的代码,我在这里判断当扫过的角度为330度的时候,也就是sweepAngle为360的时候,重绘视图,然后开启动画

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    if (isAfterAnimation) {
        sweepAngle += 30;
        if (sweepAngle >= 360) {
            isAfterAnimation = false;
            drawPicture(canvas, 360);
            startAnimation(rotateAnimation);
        } else {
            drawPicture(canvas, sweepAngle);
            postInvalidateDelayed(30);
        }
    } else {
        drawPicture(canvas, 360);
    }
}
下面是重写onMeasure方法的代码,在measureWidth方法中判断计算模式,返回空间的宽度,计算模式为wrap_content的时候宽度为200px,计算高度的方法一样,这里不再累赘;


@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    width = measureWidth(widthMeasureSpec);
    height = measureHeight(heightMeasureSpec);
    setMeasuredDimension(measureWidth(widthMeasureSpec), measureHeight(heightMeasureSpec));
}
/**
 * 计算控件宽度
 *
 * @param widthMeasureSpec
 * @return
 */
private int measureWidth(int widthMeasureSpec) {
    int result = 0;
    int mode = MeasureSpec.getMode(widthMeasureSpec);
    int size = MeasureSpec.getSize(widthMeasureSpec);

    if (mode == MeasureSpec.EXACTLY) {
        result = size;
    } else {
        result = 200;
        if (mode == MeasureSpec.AT_MOST) {
            result = Math.min(result, size);
        }
    }
    return result;
}

完整代码:

package com.example.mengsong.demo01;

/**
 * Created by MengSong on 2016/12/20.
 */

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.LinearGradient;
import android.graphics.Matrix;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.Rect;
import android.graphics.RectF;
import android.graphics.Shader;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;
import android.view.animation.AccelerateDecelerateInterpolator;
import android.view.animation.Animation;
import android.view.animation.RotateAnimation;
import android.view.animation.ScaleAnimation;
import android.widget.TextView;

/**
 * Created by MengSong on 2016/12/15.
 */
public class MyView extends View implements Animation.AnimationListener {
    //绘制三角形的路径
    private Path path;
    //绘制三角形的画笔
    public Paint mPaintTriangle;
    //绘制弧形的画笔
    public Paint mPaintArc;
    //三角形的中心到顶点的距离
    private int length = 40;
    //圆弧的半径
    private int arcRadius = 80;
    //圆弧扫过的角度
    private float sweepAngle = 360;

    private boolean isAfterAnimation = false;
    //旋转动画
    private RotateAnimation rotateAnimation;
    //圆弧所在的矩形
    private RectF rect;
    //空间的宽度
    private int width;
    //控件的高度
    private int height;

    public MyView(Context context) {
        super(context);
        init();
    }


    public MyView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    /**
     * 布局加载完成后开始动画
     */
    @Override
    protected void onFinishInflate() {
        super.onFinishInflate();
        startAnimation(rotateAnimation);
    }

    /**
     * 初始化相关属性
     */
    private void init() {
        mPaintTriangle = new Paint();
        mPaintTriangle.setDither(true);
        mPaintTriangle.setAntiAlias(true);
        mPaintTriangle.setStyle(Paint.Style.FILL);
        mPaintTriangle.setColor(Color.GREEN);

        mPaintArc = new Paint();
        mPaintArc.setDither(true);
        mPaintArc.setAntiAlias(true);
        mPaintArc.setStrokeWidth(3);
        mPaintArc.setStyle(Paint.Style.STROKE);
        mPaintArc.setColor(Color.GREEN);
        path = new Path();
        initRouteAnimation();
    }

    /**
     * 初始化三角形的path弧形的所在的矩形
     *
     * @param w
     * @param h
     * @param oldw
     * @param oldh
     */
    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);
        rect = new RectF(width / 2 - arcRadius, height / 2 - arcRadius, width / 2 + arcRadius, height / 2 + arcRadius);
        path.moveTo(width / 2, height / 2 - length);
        path.lineTo(formantNumber(width / 2 - Math.sin(Math.PI / 3) * length), width / 2 + length / 2);
        path.lineTo(formantNumber(width / 2 + Math.sin(Math.PI / 3) * length), width / 2 + length / 2);
        path.close();
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        if (isAfterAnimation) {
            sweepAngle += 30;
            if (sweepAngle >= 360) {
                isAfterAnimation = false;
                drawPicture(canvas, 360);
                startAnimation(rotateAnimation);
            } else {
                drawPicture(canvas, sweepAngle);
                postInvalidateDelayed(30);
            }
        } else {
            drawPicture(canvas, 360);
        }
    }

    private void drawPicture(Canvas canvas, float sweepAngle) {
        //画三角形
        canvas.drawPath(path, mPaintTriangle);
        //画扇形
        canvas.drawArc(rect, 270, sweepAngle, false, mPaintArc);
    }

    /**
     * 保留四位小数
     *
     * @param number
     * @return
     */
    private float formantNumber(double number) {
        java.text.DecimalFormat df = new java.text.DecimalFormat("#.####");
        String format = df.format(number);
        return Float.valueOf(format.trim());
    }

    /**
     * 动画介绍后的回调
     *
     * @param animation
     */
    @Override
    public void onAnimationEnd(Animation animation) {
        sweepAngle = 0;
        isAfterAnimation = true;
        invalidate();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        width = measureWidth(widthMeasureSpec);
        height = measureHeight(heightMeasureSpec);
        setMeasuredDimension(measureWidth(widthMeasureSpec), measureHeight(heightMeasureSpec));
    }

    /**
     * 计算控件宽度
     *
     * @param widthMeasureSpec
     * @return
     */
    private int measureWidth(int widthMeasureSpec) {
        int result = 0;
        int mode = MeasureSpec.getMode(widthMeasureSpec);
        int size = MeasureSpec.getSize(widthMeasureSpec);

        if (mode == MeasureSpec.EXACTLY) {
            result = size;
        } else {
            result = 200;
            if (mode == MeasureSpec.AT_MOST) {
                result = Math.min(result, size);
            }
        }
        return result;
    }

    /**
     * 计算控件高度
     *
     * @param heightMeasureSpec
     * @return
     */
    private int measureHeight(int heightMeasureSpec) {
        int result = 0;
        int mode = MeasureSpec.getMode(heightMeasureSpec);
        int size = MeasureSpec.getSize(heightMeasureSpec);

        if (mode == MeasureSpec.EXACTLY) {
            result = size;
        } else {
            result = 200;
            if (mode == MeasureSpec.AT_MOST) {
                result = Math.min(result, size);
            }
        }
        return result;
    }

    /**
     * 初始化动画
     */
    private void initRouteAnimation() {
        rotateAnimation = new RotateAnimation(0, 1800, Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF, 0.5f);
        rotateAnimation.setInterpolator(new AccelerateDecelerateInterpolator());
        rotateAnimation.setDuration(1500);
        rotateAnimation.setFillAfter(false);
        rotateAnimation.setAnimationListener(this);
    }

    @Override
    public void onAnimationStart(Animation animation) {
    }

    @Override
    public void onAnimationRepeat(Animation animation) {
    }
}






内容概要:该论文探讨了一种基于粒子群优化(PSO)的STAR-RIS辅助NOMA无线通信网络优化方法。STAR-RIS作为一种新型可重构智能表面,能同时反射和传输信号,与传统仅能反射的RIS不同。结合NOMA技术,STAR-RIS可以提升覆盖范围、用户容量和频谱效率。针对STAR-RIS元素众多导致获取完整信道状态信息(CSI)开销大的问题,作者提出一种在不依赖完整CSI的情况下,联合优化功率分配、基站波束成形以及STAR-RIS的传输和反射波束成形向量的方法,以最大化总可实现速率并确保每个用户的最低速率要求。仿真结果显示,该方案优于STAR-RIS辅助的OMA系统。 适合人群:具备一定无线通信理论基础、对智能反射面技术和非正交多址接入技术感兴趣的科研人员和工程师。 使用场景及目标:①适用于希望深入了解STAR-RIS与NOMA结合的研究者;②为解决无线通信中频谱资源紧张、提高系统性能提供新的思路和技术手段;③帮助理解PSO算法在无线通信优化问题中的应用。 其他说明:文中提供了详细的Python代码实现,涵盖系统参数设置、信道建模、速率计算、目标函数定义、约束条件设定、主优化函数设计及结果可视化等环节,便于读者理解和复现实验结果。此外,文章还对比了PSO与其他优化算法(如DDPG)的区别,强调了PSO在不需要显式CSI估计方面的优势。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值