热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android自定义View实现仿网易音乐唱片播放效果

这篇文章主要为大家详细介绍了Android自定义View实现仿网易音乐唱片播放效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android实现仿网易音乐唱片播放效果的具体代码,供大家参考,具体内容如下

效果图:

在values中创建attrs.xml文件

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  
       //中间图片的半径
            //图片
       //唱片旋转的速度
  

创建GramophoneView

public class GramophoneView extends View {
  /**
   * 尺寸计算设计说明:
   * 1、唱片有两个主要尺寸:中间图片的半径、黑色圆环的宽度。
   * 黑色圆环的宽度 = 图片半径的一半。
   * 2、唱针分为“手臂”和“头”,手臂分两段,一段长的一段短的,头也是一段长的一段短的。
   * 唱针四个部分的尺寸求和 = 唱片中间图片的半径+黑色圆环的宽度
   * 唱针各部分长度 比例——长的手臂:短的手臂:长的头:短的头 = 8:4:2:1
   * 3、唱片黑色圆环顶部到唱针顶端的距离 = 唱针长的手臂的长。度
   */

  private final float DEFUALT_DISK_ROTATE_SPEED = 1f;
  private final float DEFAULT_PICTURE_RADIU = 200;   // 中间图片默认半径
  private final float DEFUALT_PAUSE_NEEDLE_DEGREE = -45;   // 暂停状态时唱针的旋转角度
  private final float DEFUALT_PLAYING_NEEDLE_DEGREE = -15;  // 播放状态时唱针的旋转角度

  private int pictureRadiu;      // 中间图片的半径

  //指针
  private int smallCircleRadiu = 20;   // 唱针顶部小圆半径
  private int bigCircleRadiu = 30;    // 唱针顶部大圆半径

  private int shortArmLength;
  private int longArmleLength;     // 唱针手臂,较长那段的长度
  private int shortHeadLength;     // 唱针的头,较短那段的长度
  private int longHeadLength;
  private Paint needlePaint;

  //唱片
  private float halfMeasureWidth;
  private int diskRingWidth;      // 黑色圆环宽度
  private float diskRotateSpeed;    // 唱片旋转速度
  private Bitmap pictureBitmap;
  private Paint diskPaint;

  //状态控制
  private boolean isPlaying;
  private float currentDiskDegree;      // 唱片旋转角度
  private float currentNeddleDegree = DEFUALT_PLAYING_NEEDLE_DEGREE; // 唱针旋转角度

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

  public GramophoneView(Context context, @Nullable AttributeSet attrs) {
    super(context, attrs);
    needlePaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    diskPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.GramophoneView);

    //拿到xml中的图片和图片半径和,旋转的度数
    pictureRadiu = (int) typedArray.getDimension(R.styleable.GramophoneView_picture_radiu, DEFAULT_PICTURE_RADIU);
    diskRotateSpeed = typedArray.getFloat(R.styleable.GramophoneView_disk_rotate_speed, DEFUALT_DISK_ROTATE_SPEED);
    Drawable drawable = typedArray.getDrawable(R.styleable.GramophoneView_src);
    if (drawable == null) {
      pictureBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher);
    } else {
      pictureBitmap = ((BitmapDrawable) drawable).getBitmap();
    }

    //初始化唱片的变量
    diskRingWidth = pictureRadiu >> 1;

    shortHeadLength = (pictureRadiu + diskRingWidth) / 15;  //图片半径和黑色圆环的和 等于 指针的总长度
    lOngHeadLength= shortHeadLength <<1;  //左移相当于乘以2
    shortArmLength = longHeadLength <<1;
    lOngArmleLength= shortArmLength <<1;


  }

  /**
   * 理想的宽高是,取决于picture的 半径的
   *
   * @param widthMeasureSpec
   * @param heightMeasureSpec
   */
  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    //我们想要的理想宽高
    int width = (pictureRadiu + diskRingWidth) * 2;
    int hight = (pictureRadiu + diskRingWidth) * 2 + longArmleLength;

    //根据我们理想的宽和高 和xml中设置的宽高,按resolveSize规则做最后的取舍
    //resolveSize规则 1、精确模式,按
    int measureWidth = resolveSize(width, widthMeasureSpec);
    int measureHeight = resolveSize(hight, heightMeasureSpec);

    setMeasuredDimension(measureWidth, measureHeight);
  }

  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    halfMeasureWidth = getMeasuredWidth() >> 1;
    drawDisk(canvas);
    drawNeedle(canvas);
    if (currentNeddleDegree > DEFUALT_PAUSE_NEEDLE_DEGREE) {
      invalidate();
    }
  }

  private void drawDisk(Canvas canvas) {
    currentDiskDegree = currentDiskDegree % 360 + diskRotateSpeed;

    canvas.save();
    canvas.translate(halfMeasureWidth, longArmleLength + diskRingWidth + pictureRadiu);
    canvas.rotate(currentDiskDegree);
    diskPaint.setColor(Color.BLACK);
    diskPaint.setStyle(Paint.Style.STROKE);
    diskPaint.setStrokeWidth(pictureRadiu / 2);
    canvas.drawCircle(0, 0, pictureRadiu + diskRingWidth / 2, diskPaint);


    Path path = new Path();    // 裁剪的path路径 (为了裁剪成圆形图片,其实是将画布剪裁成了圆形)
    path.addCircle(0, 0, pictureRadiu, Path.Direction.CW);
    canvas.clipPath(path);

    Rect src = new Rect();         //将要画bitmap的那个范围
    src.set(0, 0, pictureBitmap.getWidth(), pictureBitmap.getHeight());
    Rect dst = new Rect();
    dst.set(-pictureRadiu, -pictureRadiu, pictureRadiu, pictureRadiu);   //将要将bitmap画要坐标系的那个位置
    canvas.drawBitmap(pictureBitmap, src, dst, null);
    canvas.restore();
  }

  private void drawNeedle(Canvas canvas) {
    canvas.save();

    //移动坐标原点,画指针第一段
    canvas.translate(halfMeasureWidth, 0);
    canvas.rotate(currentNeddleDegree);
    needlePaint.setColor(Color.parseColor("#C0C0C0"));
    needlePaint.setStrokeWidth(20);
    canvas.drawLine(0, 0, 0, longArmleLength, needlePaint);

    //画指针第二段
    canvas.translate(0, longArmleLength);
    canvas.rotate(-30);
    canvas.drawLine(0, 0, 0, shortArmLength, needlePaint);


    //画指针第三段
    canvas.translate(0, shortArmLength);
    needlePaint.setStrokeWidth(30);
    canvas.drawLine(0, 0, 0, longHeadLength, needlePaint);

    //画指针的第四段
    canvas.translate(0, longHeadLength);
    needlePaint.setStrokeWidth(45);
    canvas.drawLine(0, 0, 0, shortHeadLength, needlePaint);
    canvas.restore();


    //画指针的支点
    canvas.save();
    canvas.translate(halfMeasureWidth, 0);
    needlePaint.setColor(Color.parseColor("#8A8A8A"));
    needlePaint.setStyle(Paint.Style.FILL);
    canvas.drawCircle(0, 0, bigCircleRadiu, needlePaint);

    needlePaint.setColor(Color.parseColor("#C0C0C0"));
    canvas.drawCircle(0, 0, smallCircleRadiu, needlePaint);
    canvas.restore();


    //当前如果是播放的话,就移动到播放的位置 ,因为逆时针旋转度数是负的所以,- + 需要注意
    if (isPlaying) {
      if (currentNeddleDegree  DEFUALT_PAUSE_NEEDLE_DEGREE) {
        currentNeddleDegree -= 3;
      }
    }
  }

  public void pauseOrstart() {
    isPlaying = !isPlaying;
    invalidate();
  }


  /**
   * 设置图片半径
   *
   * @param pictureRadius 图片半径
   */
  public void setPictureRadius(int pictureRadius) {
    this.pictureRadiu = pictureRadius;
  }


  /**
   * 设置唱片旋转速度
   *
   * @param diskRotateSpeed 旋转速度
   */
  public void setDiskRotateSpeed(float diskRotateSpeed) {
    this.diskRotateSpeed = diskRotateSpeed;
  }

  /**
   * 设置图片资源id
   *
   * @param resId 图片资源id
   */
  public void setPictureRes(int resId) {
    pictureBitmap = BitmapFactory.decodeResource(getContext().getResources(), resId);
    invalidate();
  }
}

activity_main.xml

<&#63;xml version="1.0" encoding="utf-8"&#63;>


  

  

MainActivity

public class MainActivity extends AppCompatActivity {
  private GramophoneView gramophoneView;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    gramophOneView= (GramophoneView) findViewById(R.id.gramopone);
  }

  public void pauseOrstart(View view) {
    gramophoneView.pauseOrstart();
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • android listview OnItemClickListener失效原因
    最近在做listview时发现OnItemClickListener失效的问题,经过查找发现是因为button的原因。不仅listitem中存在button会影响OnItemClickListener事件的失效,还会导致单击后listview每个item的背景改变,使得item中的所有有关焦点的事件都失效。本文给出了一个范例来说明这种情况,并提供了解决方法。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • 本文讲述了如何通过代码在Android中更改Recycler视图项的背景颜色。通过在onBindViewHolder方法中设置条件判断,可以实现根据条件改变背景颜色的效果。同时,还介绍了如何修改底部边框颜色以及提供了RecyclerView Fragment layout.xml和项目布局文件的示例代码。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • Android系统移植与调试之如何修改Android设备状态条上音量加减键在横竖屏切换的时候的显示于隐藏
    本文介绍了如何修改Android设备状态条上音量加减键在横竖屏切换时的显示与隐藏。通过修改系统文件system_bar.xml实现了该功能,并分享了解决思路和经验。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • HDFS2.x新特性
    一、集群间数据拷贝scp实现两个远程主机之间的文件复制scp-rhello.txtroothadoop103:useratguiguhello.txt推pushscp-rr ... [详细]
  • flowable工作流 流程变量_信也科技工作流平台的技术实践
    1背景随着公司业务发展及内部业务流程诉求的增长,目前信息化系统不能够很好满足期望,主要体现如下:目前OA流程引擎无法满足企业特定业务流程需求,且移动端体 ... [详细]
  • 本文介绍了在使用MSXML解析XML文件时出现DTD禁用问题的解决方案。通过代码示例和错误信息获取方法,解释了默认情况下DTD是禁用的,以及如何启用DTD的方法。此外,还提到了网上关于该问题的信息相对较少,因此本文提供了解决方案以供参考。 ... [详细]
  • 在Xamarin XAML语言中如何在页面级别构建ControlTemplate控件模板
    本文介绍了在Xamarin XAML语言中如何在页面级别构建ControlTemplate控件模板的方法和步骤,包括将ResourceDictionary添加到页面中以及在ResourceDictionary中实现模板的构建。通过本文的阅读,读者可以了解到在Xamarin XAML语言中构建控件模板的具体操作步骤和语法形式。 ... [详细]
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
author-avatar
yf_992258
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有