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

Android实现悬浮图片

这篇文章主要为大家详细介绍了Android实现悬浮图片,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Android实现悬浮图片的具体代码,供大家参考,具体内容如下

@SuppressLint("AppCompatCustomView")
public class MoveImageView extends ImageView {

 //按下那一刻的坐标和 控件上下左右距离
 private float lastX;
 private float lastY;
 private int left;
 private int top;
 private int right;
 private int bottom;


 //如果是拖动事件就不用响应点击事件
 boolean isMove = false;
 boolean isAnimatoring = false;

 //屏幕宽高
 private int screenWidthPx;
 private int screenHeightPx;


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

 public MoveImageView(Context context, @Nullable AttributeSet attrs) {
  this(context, attrs, 0);
 }

 public MoveImageView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
  super(context, attrs, defStyleAttr);
  screenWidthPx = getScreenWidthPx(getContext());
  screenHeightPx = getScreenHeightPx(getContext());
 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
  switch (event.getAction()) {
   case MotionEvent.ACTION_DOWN:
    //动画执行过程中,不响应一切操作,
    // 这里相当于不让其走后面的MotionEvent.ACTION_MOVE:与MotionEvent.ACTION_UP:
    if (isAnimatoring) {
     return false;
    }
    lastX = event.getRawX();
    lastY = event.getRawY();
    left = getLeft();
    top = getTop();
    right = getRight();
    bottom = getBottom();
    break;
   case MotionEvent.ACTION_MOVE:
    isMove = true;
    float x = event.getRawX();
    float y = event.getRawY();


    int l = (int) (left + (x - lastX));
    int t = (int) (top + (y - lastY));
    int r = (int) (right + (x - lastX));
    int b = (int) (bottom + (y - lastY));
    layout(l, t, r, b);
    break;
   case MotionEvent.ACTION_UP:
    if (isMove) {
     //如果顶部拖出屏幕外面,回正
     if (getTop() <0) {
      layout(getLeft(), 0, getRight(), getHeight());
     }

     //getBottom() 获取到的是 控件底部到父容器顶部的距离,所以需要减去状态栏的高度
     int bottomHeight = screenHeightPx - getStatusBarHeight(getContext());
     //如果底部拖出屏幕外面,回正
     if (getBottom() > bottomHeight) {
      layout(getLeft(), bottomHeight-getHeight(), getRight(), bottomHeight);
     }
     isMove = false;
     startAnimation();
     return true;
    }
    return super.onTouchEvent(event);
  }
  return super.onTouchEvent(event);
 }

 private void startAnimation() {

  isAnimatoring = true;

  //右边距
  int marinRight = DisplayUtils.dpToPx(20);
  int endValue = screenWidthPx - marinRight;
  ValueAnimator animator = ValueAnimator.ofInt(getRight(), endValue);

  animator.setDuration(Math.abs(endValue - getRight()) > 1000 &#63; 1000 : Math.abs(endValue - getRight()));

  animator.addUpdateListener(new ValueAnimator.AnimatorUpdateListener() {
   @Override
   public void onAnimationUpdate(ValueAnimator animation) {
    int curValue = (int) animation.getAnimatedValue();
    layout(curValue - getWidth(), getTop(), curValue, getHeight() + getTop());
   }
  });
  animator.addListener(new AnimatorListenerAdapter() {
   @Override
   public void onAnimationStart(Animator animation) {

   }

   @Override
   public void onAnimationEnd(Animator animation) {
    isAnimatoring = false;
    animator.removeAllUpdateListeners();
    animator.removeAllListeners();
   }
  });
  animator.start();
 }

 /**
  * 获取状态栏高度
  */
 public static int getStatusBarHeight(Context context) {
  int result = 24;
  int resId = context.getResources().getIdentifier("status_bar_height", "dimen", "android");
  if (resId > 0) {
   result = context.getResources().getDimensionPixelSize(resId);
  } else {
   result = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP,
     result, Resources.getSystem().getDisplayMetrics());
  }
  return result;
 }

 public static int getScreenWidthPx(Context context) {
  WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
  DisplayMetrics dm = new DisplayMetrics();
  if (windowManager != null) {
//   windowManager.getDefaultDisplay().getMetrics(dm);
   windowManager.getDefaultDisplay().getRealMetrics(dm);
   return dm.widthPixels;
  }
  return 0;

 }

 public static int getScreenHeightPx(Context context) {
  WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
  DisplayMetrics dm = new DisplayMetrics();
  if (windowManager != null) {
//   windowManager.getDefaultDisplay().getMetrics(dm);
   windowManager.getDefaultDisplay().getRealMetrics(dm);
   return dm.heightPixels;
  }
  return 0;

 }

}

应用:

布局文件:

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



 


Activity:

public class TestActivity extends AppCompatActivity {

 private MoveImageView moveImageView;

 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_test2);
  moveImageView = ((MoveImageView) findViewById(R.id.MoveImageView));
  moveImageView.setOnClickListener(new View.OnClickListener() {
   @Override
   public void onClick(View v) {
    Toast.makeText(TestActivity.this, "点击事件", Toast.LENGTH_SHORT).show();
   }
  });
 }
}

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


推荐阅读
  • 深入解析Android自定义View面试题
    本文探讨了Android Launcher开发中自定义View的重要性,并通过一道经典的面试题,帮助开发者更好地理解自定义View的实现细节。文章不仅涵盖了基础知识,还提供了实际操作建议。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍了如何使用Spring Boot进行高效开发,涵盖了配置、实例化容器以及核心注解的使用方法。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 本文介绍如何在 Unity 的 XML 配置文件中,将参数传递给自定义生命周期管理器的构造函数。我们将详细探讨 CustomLifetimeManager 类的实现及其配置方法。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 探讨如何真正掌握Java EE,包括所需技能、工具和实践经验。资深软件教学总监李刚分享了对毕业生简历中常见问题的看法,并提供了详尽的标准。 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
  • Startup 类配置服务和应用的请求管道。Startup类ASP.NETCore应用使用 Startup 类,按照约定命名为 Startup。 Startup 类:可选择性地包括 ... [详细]
  • 本文介绍了Android开发中Intent的基本概念及其在不同Activity之间的数据传递方式,详细展示了如何通过Intent实现Activity间的跳转和数据传输。 ... [详细]
  • 本文介绍了多个关于JavaScript的书籍资源、实用工具和编程实例,涵盖从入门到进阶的各个阶段,帮助读者全面提升JavaScript编程能力。 ... [详细]
  • 科研单位信息系统中的DevOps实践与优化
    本文探讨了某科研单位通过引入云原生平台实现DevOps开发和运维一体化,显著提升了项目交付效率和产品质量。详细介绍了如何在实际项目中应用DevOps理念,解决了传统开发模式下的诸多痛点。 ... [详细]
  • 本文详细介绍了 Flink 和 YARN 的交互机制。YARN 是 Hadoop 生态系统中的资源管理组件,类似于 Spark on YARN 的配置方式。我们将基于官方文档,深入探讨如何在 YARN 上部署和运行 Flink 任务。 ... [详细]
author-avatar
暖倪2502938803
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有