Android悬浮窗的简单实现(2)

现在很多应用都有小悬浮窗的功能,比如看直播的时候,通过Home键返回桌面,直播的小窗口仍可以在屏幕上显示。下面将介绍下悬浮窗的的一种简单实现方式。

2.原理

Window我们应该很熟悉,它是一个接口类,具体的实现类为PhoneWindow,它可以对View进行管理。WindowManager是一个接口类,继承自ViewManager,从名称就知道它是用来管理Window的,它的实现类是WindowManagerImpl。如果我们想要对Window(View)进行添加、更新和删除操作就可以使用WindowManagerWindowManager会将具体的工作交由WindowManagerService处理。这里我们只需要知道WindowManager能用来管理Window就好。

WindowManager是一个接口类,继承自ViewManagerViewManager中定义了3个方法,分布用来添加、更新和删除View,如下所示:

public interface ViewManager {

public void addView(View view, ViewGroup.LayoutParams params);

public void updateViewLayout(View view, ViewGroup.LayoutParams params);

public void removeView(View view);

}

WindowManager也继承了这些方法,而这些方法传入的参数都是View类型,说明了Window是以View的形式存在的。

3.具体实现

3.1浮窗布局

悬浮窗的简易布局如下的可参考下面的layout_floating_window.xml文件。顶层深色部分的FrameLayout布局是用来实现悬浮窗的拖拽功能的,点击右上角ImageView可以实现关闭悬浮窗,剩下区域显示内容,这里只是简单地显示文本内容,不做复杂的东西,故只设置TextView。

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout

xmlns:android=“http://schemas.android.com/apk/res/android”

xmlns:app=“http://schemas.android.com/apk/res-auto”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:orientation=“vertical”>

<FrameLayout

android:id=“@+id/layout_drag”

android:layout_width=“match_parent”

android:layout_height=“15dp”

android:background=“#dddddd”>

<androidx.appcompat.widget.AppCompatImageView

android:id=“@+id/iv_close”

android:layout_width=“15dp”

android:layout_height=“15dp”

android:layout_gravity=“end”

android:src=“@drawable/img_delete”/>

<androidx.appcompat.widget.AppCompatTextView

android:id=“@+id/tv_content”

android:layout_width=“match_parent”

android:layout_height=“match_parent”

android:layout_gravity=“center_horizontal”

android:background=“#eeeeee”

android:scrollbars=“vertical”/>

3.2 悬浮窗的实现

1. 使用服务Service

Service 是一种可在后台执行长时间运行操作而不提供界面的应用组件,可由其他应用组件启动,而且即使用户切换到其他应用,仍将在后台继续运行。要保证应用在后台时,悬浮窗仍然可以正常显示,所以这里可以使用Service

2. 获取WindowManager并设置LayoutParams

private lateinit var windowManager: WindowManager

private lateinit var layoutParams: WindowManager.LayoutParams

override fun onCreate() {

// 获取WindowManager

windowManager = getSystemService(WINDOW_SERVICE) as WindowManager

layoutParams = WindowManager.LayoutParams().apply {

// 实现在其他应用和窗口上方显示浮窗

type = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {

WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY

} else {

WindowManager.LayoutParams.TYPE_PHONE

}

format = PixelFormat.RGBA_8888

// 设置浮窗的大小和位置

gravity = Gravity.START or Gravity.TOP

flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL or WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE

width = 600

height = 600

x = 300

y = 300

}

}

3. 创建View并添加到WindowManager

private lateinit var floatingView: View

override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {

if (Settings.canDrawOverlays(this)) {

floatingView = LayoutInflater.from(this).inflate(R.layout.layout_floating_window.xml, null)

windowManager.addView(floatingView, layoutParams)

}

return super.onStartCommand(intent, flags, startId)

}

4. 实现悬浮窗的拖拽和关闭功能

// 浮窗的坐标

private var x = 0

private var y = 0

override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {

if (Settings.canDrawOverlays(this)) {

floatingView = LayoutInflater.from(this).inflate(R.layout.layout_floating_window.xml, null)

windowManager.addView(floatingView, layoutParams)

// 点击浮窗的右上角关闭按钮可以关闭浮窗

floatingView.findViewById(R.id.iv_close).setOnClickListener {

windowManager.removeView(floatingView)

}

// 实现浮窗的拖动功能, 通过改变layoutParams来实现

floatingView.findViewById(R.id.layout_drag).setOnTouchListener { v, event ->

when (event.action) {

MotionEvent.ACTION_DOWN -> {

x = event.rawX.toInt()

y = event.rawY.toInt()

}

MotionEvent.ACTION_MOVE -> {

val currentX = event.rawX.toInt()

val currentY = event.rawY.toInt()

val offsetX = currentX - x

val offsetY = currentY - y

x = currentX

y = currentY

layoutParams.x = layoutParams.x + offsetX

layoutParams.y = layoutParams.y + offsetY

// 更新floatingView

windowManager.updateViewLayout(floatingView, layoutParams)

}

}

true

}

return super.onStartCommand(intent, flags, startId)

}

5. 利用广播进行通信

private var receiver: MyReceiver? = null

override fun onCreate() {

// 注册广播

receiver = MyReceiver()

val filter = IntentFilter()

filter.addAction(“android.intent.action.MyReceiver”)

registerReceiver(receiver, filter)

}

inner class MyReceiver : BroadcastReceiver() {

override fun onReceive(context: Context, intent: Intent) {

val content = intent.getStringExtra(“content”) ?: “”

// 通过Handler更新UI

val message = Message.obtain()

message.what = 0

message.obj = content

handler.sendMessage(message)

}

}

val handler = Handler(this.mainLooper) { msg ->

tvContent.text = msg.obj as String

false

}

可以在Activity中通过广播给Service发送信息

fun sendMessage(view: View?) {

Intent(“android.intent.action.MyReceiver”).apply {

putExtra(“content”, “Hello, World!”)

sendBroadcast(this)

}

}

6. 设置权限

悬浮窗的显示需要权限,在AndroidManefest.xml中添加:

此外,还要通过Settings.ACTION_MANAGE_OVERLAY_PERMISSION来让动态设置权限,在Activity中设置。

// MainActivity.kt

fun startWindow(view: View?) {

if (!Settings.canDrawOverlays(this)) {

startActivityForResult(Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse(“package:$packageName”)), 0)

} else {

startService(Intent(this@MainActivity, FloatingWindowService::class.java))

}

}

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {

super.onActivityResult(requestCode, resultCode, data)

if (requestCode == 0) {

if (Settings.canDrawOverlays(this)) {

Toast.makeText(this, “悬浮窗权限授权成功”, Toast.LENGTH_SHORT).show()

startService(Intent(this@MainActivity, FloatingWindowService::class.java))

}

}

}

3.3 完整代码

class FloatingWindowService : Service() {

private lateinit var windowManager: WindowManager

private lateinit var layoutParams: WindowManager.LayoutParams

private lateinit var tvContent: AppCompatTextView

private lateinit var handler: Handler

private var receiver: MyReceiver? = null

private var floatingView: View? = null

private val stringBuilder = StringBuilder()

private var x = 0

private var y = 0

最后

最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2019-2021BAT 面试真题解析,我把大厂面试中常被问到的技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。

还有 高级架构技术进阶脑图 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

Android 基础知识点

Java 基础知识点

Android 源码相关分析

常见的一些原理性问题

希望大家在今年一切顺利,进到自己想进的公司,共勉!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

最后

最后为了帮助大家深刻理解Android相关知识点的原理以及面试相关知识,这里放上相关的我搜集整理的24套腾讯、字节跳动、阿里、百度2019-2021BAT 面试真题解析,我把大厂面试中常被问到的技术点整理成了视频和PDF(实际上比预期多花了不少精力),包知识脉络 + 诸多细节。

还有 高级架构技术进阶脑图 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

Android 基础知识点

Java 基础知识点

Android 源码相关分析

常见的一些原理性问题

[外链图片转存中…(img-dFSnIRiF-1714550733888)]

希望大家在今年一切顺利,进到自己想进的公司,共勉!
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 16
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,关于 Android广告的实现,需要用到 Android 的 WindowManager 类,下面是一个简单实现方式: 1. 在 AndroidManifest.xml 文件中添加权限: ``` <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" /> ``` 2. 创建一个悬浮的布局文件,例如 float_window.xml: ```xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/float_layout" android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="@android:color/transparent" android:orientation="vertical"> <!-- 在这里添加你的广告布局 --> </LinearLayout> ``` 3. 创建一个 Service 类用于控制悬浮的显示和隐藏: ```java public class FloatWindowService extends Service { private WindowManager windowManager; private LinearLayout floatLayout; private WindowManager.LayoutParams params; @Override public void onCreate() { super.onCreate(); windowManager = (WindowManager) getSystemService(WINDOW_SERVICE); // 设置悬浮的布局参数 params = new WindowManager.LayoutParams( WindowManager.LayoutParams.WRAP_CONTENT, WindowManager.LayoutParams.WRAP_CONTENT, WindowManager.LayoutParams.TYPE_SYSTEM_ALERT, WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, PixelFormat.TRANSLUCENT); // 设置悬浮的位置 params.gravity = Gravity.TOP | Gravity.LEFT; params.x = 0; params.y = 0; // 加载悬浮的布局文件 LayoutInflater inflater = LayoutInflater.from(this); floatLayout = (LinearLayout) inflater.inflate(R.layout.float_window, null); // 在这里添加你的广告布局 // 将悬浮添加到 WindowManager 中 windowManager.addView(floatLayout, params); } @Override public void onDestroy() { super.onDestroy(); // 在 Service 销毁时移除悬浮 if (floatLayout != null) { windowManager.removeView(floatLayout); } } @Nullable @Override public IBinder onBind(Intent intent) { return null; } } ``` 4. 在需要显示悬浮的地方启动 Service: ```java startService(new Intent(this, FloatWindowService.class)); ``` 这样就可以实现一个简单Android广告了。当然,你也可以根据自己的需求对悬浮进行更加复杂的实现

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值