Android自定义viewGroup实现点击动画效果

作者:大城小小爱 时间:2022-09-01 00:27:07 

本文实例为大家分享了viewGroup实现点击动画效果展示的具体代码,供大家参考,具体内容如下


public class MyCustomView extends ViewGroup implements View.OnClickListener {

private OnMenuItemClickListener mMenuItemClickListener;

/**
  * 点击子菜单项的回调接口
  */
 public interface OnMenuItemClickListener {
   void onClick(View view, int pos);
 }

public void setOnMenuItemClickListener(
     OnMenuItemClickListener mMenuItemClickListener) {
   this.mMenuItemClickListener = mMenuItemClickListener;
 }

public enum Status {
   OPEN, CLOSE
 }

private int mRadius;
 /**
  * 菜单的状态
  */
 private Status mCurrentStatus = Status.CLOSE;
 /**
  * 菜单的主按钮
  */
 private View mCButton;

public MyCustomView(Context context) {//通过new对象来调用
   this(context, null);
   Log.i("jj", "super(context)");
 }

public MyCustomView(Context context, AttributeSet attrs) {//在布局中使用时调用
   this(context, attrs, 0);
   Log.i("jj", "super(context, attrs)");
 }

public MyCustomView(Context context, AttributeSet attrs, int defStyleAttr) {
   super(context, attrs, defStyleAttr);
   Log.i("jj", "super(context, attrs, defStyleAttr)");
   mRadius = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 100, getResources().getDisplayMetrics());
   Log.i("jj", "mRadius1: " + mRadius);
   // 获取自定义属性的值
   TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.MyCustomView, defStyleAttr, 0);
   mRadius = (int) a.getDimension(R.styleable.MyCustomView_radius, TypedValue
       .applyDimension(TypedValue.COMPLEX_UNIT_DIP, 100,
           getResources().getDisplayMetrics()));
   Log.i("jj", "mRadius: " + mRadius);
   a.recycle();
 }

@Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
   int count = getChildCount();
   for (int i = 0; i < count; i++) {
     // 测量child
     measureChild(getChildAt(i), widthMeasureSpec, heightMeasureSpec);
   }
   super.onMeasure(widthMeasureSpec, heightMeasureSpec);
 }

@Override
 protected void onLayout(boolean changed, int l, int t, int r, int b) {
   if (changed) {
     layoutCButton();
     int count = getChildCount();
     for (int i = 0; i < count - 1; i++) {
       View child = getChildAt(i + 1);
       child.setVisibility(View.GONE);
       int cl = (int) (mRadius * Math.sin(Math.PI / 2 / (count - 2)
           * i));
       int ct = (int) (mRadius * Math.cos(Math.PI / 2 / (count - 2)
           * i));
       int cWidth = child.getMeasuredWidth();
       int cHeight = child.getMeasuredHeight();
       ct = getMeasuredHeight() - cHeight - ct;
       cl = getMeasuredWidth() - cWidth - cl;
       child.layout(cl, ct, cl + cWidth, ct + cHeight);
     }
   }
 }

private void layoutCButton() {
   mCButton = getChildAt(0);
   mCButton.setOnClickListener(this);
   int width = mCButton.getMeasuredWidth();
   int height = mCButton.getMeasuredHeight();
   int l = getMeasuredWidth() - width;
   int t = getMeasuredHeight() - height;
   mCButton.layout(l, t, l + width, t + width);
 }

@Override
 public void onClick(View v) {
   rotateCButton(v, 0f, 360f, 300);
   toggleMenu(300);
 }

private void rotateCButton(View v, float start, float end, int duration) {

RotateAnimation anim = new RotateAnimation(start, end,
       Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF,
       0.5f);
   anim.setDuration(duration);
   anim.setFillAfter(true);
   v.startAnimation(anim);
 }

/**
  * 切换菜单
  */
 public void toggleMenu(int duration) {
   // 为menuItem添加平移动画和旋转动画
   int count = getChildCount();
   for (int i = 0; i < count - 1; i++) {
     final View childView = getChildAt(i + 1);
     childView.setVisibility(View.VISIBLE);
     int cl = (int) (mRadius * Math.sin(Math.PI / 2 / (count - 2) * i));
     int ct = (int) (mRadius * Math.cos(Math.PI / 2 / (count - 2) * i));

AnimationSet animset = new AnimationSet(true);
     Animation tranAnim = null;
     int xflag = 1;
     int yflag = 1;
     // to open
     if (mCurrentStatus == Status.CLOSE) {
       tranAnim = new TranslateAnimation(xflag * cl, 0, yflag * ct, 0);
       childView.setClickable(true);
       childView.setFocusable(true);

} else
     // to close
     {
       tranAnim = new TranslateAnimation(0, xflag * cl, 0, yflag * ct);
       childView.setClickable(false);
       childView.setFocusable(false);
     }
     tranAnim.setFillAfter(true);
     tranAnim.setDuration(duration);
     tranAnim.setStartOffset((i * 100) / count);
     tranAnim.setAnimationListener(new Animation.AnimationListener() {

@Override
       public void onAnimationStart(Animation animation) {

}

@Override
       public void onAnimationRepeat(Animation animation) {

}

@Override
       public void onAnimationEnd(Animation animation) {
         if (mCurrentStatus == Status.CLOSE) {
           childView.setVisibility(View.GONE);
         }
       }
     });
     // 旋转动画
     RotateAnimation rotateAnim = new RotateAnimation(0, 720,
         Animation.RELATIVE_TO_SELF, 0.5f,
         Animation.RELATIVE_TO_SELF, 0.5f);
     rotateAnim.setDuration(duration);
     rotateAnim.setFillAfter(true);
     animset.addAnimation(rotateAnim);
     animset.addAnimation(tranAnim);
     childView.startAnimation(animset);
     final int pos = i + 1;
     childView.setOnClickListener(new OnClickListener() {
       @Override
       public void onClick(View v) {
         if (mMenuItemClickListener != null)
           mMenuItemClickListener.onClick(childView, pos);
         menuItemAnim(pos - 1);
         changeStatus();

}
     });
   }
   // 切换菜单状态
   changeStatus();
 }

/**
  * 添加menuItem的点击动画
  *
  * @param
  */
 private void menuItemAnim(int pos) {
   for (int i = 0; i < getChildCount() - 1; i++) {

View childView = getChildAt(i + 1);
     if (i == pos) {
       childView.startAnimation(scaleBigAnim(300));
     } else {
       childView.startAnimation(scaleSmallAnim(300));
     }

childView.setClickable(false);
     childView.setFocusable(false);

}

}

private Animation scaleSmallAnim(int duration) {

AnimationSet animationSet = new AnimationSet(true);

ScaleAnimation scaleAnim = new ScaleAnimation(1.0f, 0.0f, 1.0f, 0.0f,
       Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF,
       0.5f);
   AlphaAnimation alphaAnim = new AlphaAnimation(1f, 0.0f);
   animationSet.addAnimation(scaleAnim);
   animationSet.addAnimation(alphaAnim);
   animationSet.setDuration(duration);
   animationSet.setFillAfter(true);
   return animationSet;

}

/**
  * 为当前点击的Item设置变大和透明度降低的动画
  *
  * @param duration
  * @return
  */
 private Animation scaleBigAnim(int duration) {
   AnimationSet animationSet = new AnimationSet(true);

ScaleAnimation scaleAnim = new ScaleAnimation(1.0f, 4.0f, 1.0f, 4.0f,
       Animation.RELATIVE_TO_SELF, 0.5f, Animation.RELATIVE_TO_SELF,
       0.5f);
   AlphaAnimation alphaAnim = new AlphaAnimation(1f, 0.0f);

animationSet.addAnimation(scaleAnim);
   animationSet.addAnimation(alphaAnim);

animationSet.setDuration(duration);
   animationSet.setFillAfter(true);
   return animationSet;

}

public boolean isOpen() {
   return mCurrentStatus == Status.OPEN;
 }

/**
  * 切换菜单状态
  */
 private void changeStatus() {
   mCurrentStatus = (mCurrentStatus == Status.CLOSE ? Status.OPEN
       : Status.CLOSE);
 }

}

attrs.xml:


<?xml version="1.0" encoding="utf-8"?>
<resources>
 <attr name="radius" format="dimension" />
 <declare-styleable name="MyCustomView">
   <attr name="radius" />
 </declare-styleable>
</resources>

菜单布局文件:


<com.admom.mycanvas.view.MyCustomView xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:hyman="http://schemas.android.com/apk/res-auto"
 xmlns:tools="http://schemas.android.com/tools"
 android:id="@+id/id_menu"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 hyman:radius="160dp">

<RelativeLayout
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:background="@drawable/composer_button">

<ImageView
     android:id="@+id/id_button"
     android:layout_width="wrap_content"
     android:layout_height="wrap_content"
     android:layout_centerInParent="true"
     android:src="@drawable/composer_icn_plus" />
 </RelativeLayout>

<ImageView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:src="@drawable/composer_music"
   android:tag="Music" />

<ImageView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:src="@drawable/composer_place"
   android:tag="Place" />

<ImageView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:src="@drawable/composer_sleep"
   android:tag="Sleep" />

<ImageView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:src="@drawable/composer_thought"
   android:tag="Sun" />

<ImageView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:src="@drawable/composer_with"
   android:tag="People" />

</com.admom.mycanvas.view.MyCustomView>

主界面布局文件:


<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:layout_width="match_parent" android:layout_height="match_parent">

<include layout="@layout/menu_right_bottom"/>

</RelativeLayout>

在主程序直接使用:


mMenu = (MyCustomView) findViewById(R.id.id_menu);
mMenu.setOnMenuItemClickListener(new MyCustomView.OnMenuItemClickListener() {
 @Override
 public void onClick(View view, int pos) {
   Toast.makeText(MainActivity.this, pos + ":" + view.getTag(), Toast.LENGTH_SHORT).show();
 }
});
mMenu.setOnTouchListener(new View.OnTouchListener() {
 @Override
 public boolean onTouch(View v, MotionEvent event) {
   if(mMenu.isOpen()){
     mMenu.toggleMenu(300);
   }
   return false;
 }
});

来源:http://blog.csdn.net/Scorpio_gao/article/details/53153788

标签:Android,viewGroup,点击动画
0
投稿

猜你喜欢

  • Java快速批量移动文件的实现方法

    2021-06-15 16:58:56
  • C语言根据协议分割获取字符串单元的实现代码

    2023-06-21 08:20:27
  • Spring Boot应用监控的实战教程

    2022-03-02 18:17:09
  • C#中的除法运算符与VB.NET中的除法运算符

    2022-04-01 10:52:56
  • SpringBoot整合Quartz实现定时任务详解

    2021-11-22 06:40:57
  • 使用EasyPoi轻松导入导出Excel文档的方法示例

    2023-12-26 11:47:56
  • c#读写App.config,ConfigurationManager.AppSettings 不生效的解决方法

    2021-10-07 22:34:42
  • C# 常用协议实现模版及FixedSizeReceiveFilter示例(SuperSocket入门)

    2023-12-12 04:17:48
  • C# 调用Delphi dll 实例代码

    2023-06-12 22:46:06
  • java连接ElasticSearch集群操作

    2023-11-28 04:06:24
  • android 自定义圆角button效果的实例代码(自定义view Demo)

    2022-07-13 11:35:09
  • Spring如何利用@Value注解读取yml中的map配置

    2023-07-24 21:18:00
  • Android EditText限制输入字符类型的方法总结

    2023-10-11 15:48:19
  • Struts2中异常处理机制分析

    2023-11-17 21:54:32
  • 一篇文章带你复习java知识点

    2023-01-15 09:39:38
  • C# 如何规范的写 DEBUG 输出

    2023-04-27 03:12:37
  • 深入浅析SpringBoot自动配置原理

    2022-11-19 01:49:24
  • java基础入门之IO流

    2022-08-17 00:09:20
  • java实现文件归档和还原

    2023-02-28 23:09:51
  • Java反射概念与使用实例代码

    2022-02-24 23:55:40
  • asp之家 软件编程 m.aspxhome.com