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

// 用来判断floatingView是否attached 到 window manager,防止二次removeView导致崩溃

private var attached = false

override fun onCreate() {

super.onCreate()

// 注册广播

receiver = MyReceiver()

val filter = IntentFilter()

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

registerReceiver(receiver, filter);

// 获取windowManager并设置layoutParams

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

// format = PixelFormat.TRANSPARENT

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

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

最后

题外话,我在一线互联网企业工作十余年里,指导过不少同行后辈。帮助很多人得到了学习和成长。

我意识到有很多经验和知识值得分享给大家,也可以通过我们的能力和经验解答大家在IT学习中的很多困惑,所以在工作繁忙的情况下还是坚持各种整理和分享。但苦于知识传播途径有限,很多程序员朋友无法获得正确的资料得到学习提升,故此将并将重要的Android进阶资料包括自定义view、性能优化、MVC与MVP与MVVM三大框架的区别、NDK技术、阿里面试题精编汇总、常见源码分析等学习资料。

【Android思维脑图(技能树)】

知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。

Android开发8年,阿里、百度一面惨被吊打!我是否应该转行了?

【Android进阶学习视频】、【全套Android面试秘籍】

希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展

知识值得分享给大家,也可以通过我们的能力和经验解答大家在IT学习中的很多困惑,所以在工作繁忙的情况下还是坚持各种整理和分享。但苦于知识传播途径有限,很多程序员朋友无法获得正确的资料得到学习提升,故此将并将重要的Android进阶资料包括自定义view、性能优化、MVC与MVP与MVVM三大框架的区别、NDK技术、阿里面试题精编汇总、常见源码分析等学习资料。

【Android思维脑图(技能树)】

知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。

[外链图片转存中…(img-LPcAuwOe-1711738857859)]

【Android进阶学习视频】、【全套Android面试秘籍】

希望我能够用我的力量帮助更多迷茫、困惑的朋友们,帮助大家在IT道路上学习和发展

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

Logo

这里是“一人公司”的成长家园。我们提供从产品曝光、技术变现到法律财税的全栈内容,并连接云服务、办公空间等稀缺资源,助你专注创造,无忧运营。

更多推荐