版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
【移動(dòng)應(yīng)用開發(fā)技術(shù)】Android懸浮窗如何實(shí)現(xiàn)
顯示浮窗public
interface
ViewManager{
//'向窗口添加視圖'
public
void
addView(View
view,
ViewGroup.LayoutParams
params);
//'更新窗口中視圖'
public
void
updateViewLayout(View
view,
ViewGroup.LayoutParams
params);
//'移除窗口中視圖'
public
void
removeView(View
view);
}
復(fù)制代碼//'解析布局文件為視圖'val
windowView
=
LayoutInflater.from(context).inflate(R.id.window_view,
null)//'獲取WindowManager系統(tǒng)服務(wù)'val
windowManager
=
context.getSystemService(Context.WINDOW_SERVICE)
as
WindowManager//'構(gòu)建窗口布局參數(shù)'WindowManager.LayoutParams().apply
{
type
=
WindowManager.LayoutParams.TYPE_APPLICATION
width
=
WindowManager.LayoutParams.WRAP_CONTENT
height
=
WindowManager.LayoutParams.WRAP_CONTENT
gravity
=
Gravity.START
or
Gravity.TOP
x
=
0
y
=
0}.let
{
layoutParams->
//'將視圖添加到窗口'
windowManager.addView(windowView,
layoutParams)
}
復(fù)制代碼object
FloatWindow{
private
var
context:
Context?
=
null
//'當(dāng)前窗口參數(shù)'
var
windowInfo:
WindowInfo?
=
null
//'把和Window布局有關(guān)的參數(shù)打包成一個(gè)內(nèi)部類'
class
WindowInfo(var
view:
View?)
{
var
layoutParams:
WindowManager.LayoutParams?
=
null
//'窗口寬'
var
width:
Int
=
0
//'窗口高'
var
height:
Int
=
0
//'窗口中是否有視圖'
fun
hasView()
=
view
!=
null
&&
layoutParams
!=
null
//'窗口中視圖是否有父親'
fun
hasParent()
=
hasView()
&&
view?.parent
!=
null
}
//'顯示窗口'
fun
show(
context:
Context,
windowInfo:
WindowInfo?,
x:
Int
=
windowInfo?.layoutParams?.x.value(),
y:
Int
=
windowInfo?.layoutParams?.y.value(),
)
{
if
(windowInfo
==
null)
{
return
}
if
(windowInfo.view
==
null)
{
return
}
this.windowInfo
=
windowInfo
this.context
=
context
//'創(chuàng)建窗口布局參數(shù)'
windowInfo.layoutParams
=
createLayoutParam(x,
y)
//'顯示窗口'
if
(!windowInfo.hasParent().value())
{
val
windowManager
=
this.context?.getSystemService(Context.WINDOW_SERVICE)
as
WindowManager
windowManager.addView(windowInfo.view,
windowInfo.layoutParams)
}
}
//'創(chuàng)建窗口布局參數(shù)'
private
fun
createLayoutParam(x:
Int,
y:
Int):
WindowManager.LayoutParams
{
if
(context
==
null)
{
return
WindowManager.LayoutParams()
}
return
WindowManager.LayoutParams().apply
{
//'該類型不需要申請(qǐng)權(quán)限'
type
=
WindowManager.LayoutParams.TYPE_APPLICATION
format
=
PixelFormat.TRANSLUCENT
flags
=
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
or
WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
gravity
=
Gravity.START
or
Gravity.TOP
width
=
windowInfo?.width.value()
height
=
windowInfo?.height.value()
this.x
=
x
this.y
=
y
}
}
//'為空Int提供默認(rèn)值'
fun
Int?.value()
=
this
?:
0}
復(fù)制代碼val
windowView
=
LayoutInflater.from(context).inflate(R.id.window_view,
null)
WindowInfo(windowView).apply{
width
=
100
height
=
100
}.let{
windowInfo
->
FloatWindow.show(context,
windowInfo,
0,
0)
}
復(fù)制代碼浮窗背景色object
FloatWindow{
//當(dāng)前窗口參數(shù)
var
windowInfo:
WindowInfo?
=
null
private
fun
createLayoutParam(x:
Int,
y:
Int):
WindowManager.LayoutParams
{
if
(context
==
null)
{
return
WindowManager.LayoutParams()
}
return
WindowManager.LayoutParams().apply
{
type
=
WindowManager.LayoutParams.TYPE_APPLICATION
format
=
PixelFormat.TRANSLUCENT
flags
=
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
or
WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
or
//'設(shè)置浮窗背景變暗'
WindowManager.LayoutParams.FLAG_DIM_BEHIND
//'設(shè)置默認(rèn)變暗程度為0,即不變暗,1表示全黑'
dimAmount
=
0f
gravity
=
Gravity.START
or
Gravity.TOP
width
=
windowInfo?.width.value()
height
=
windowInfo?.height.value()
this.x
=
x
this.y
=
y
}
}
//'供業(yè)務(wù)界面在需要的時(shí)候調(diào)整浮窗背景亮暗'
fun
setDimAmount(amount:Float){
windowInfo?.layoutParams?.let
{
it.dimAmount
=
amount
}
}
}
復(fù)制代碼設(shè)置浮窗點(diǎn)擊事件object
FloatWindow
:
View.OnTouchListener{
//'顯示窗口'
fun
show(
context:
Context,
windowInfo:
WindowInfo?,
x:
Int
=
windowInfo?.layoutParams?.x.value(),
y:
Int
=
windowInfo?.layoutParams?.y.value(),
)
{
if
(windowInfo
==
null)
{
return
}
if
(windowInfo.view
==
null)
{
return
}
this.windowInfo
=
windowInfo
this.context
=
context
//'為浮窗視圖設(shè)置觸摸監(jiān)聽器'
windowInfo.view?.setOnTouchListener(this)
windowInfo.layoutParams
=
createLayoutParam(x,
y)
if
(!windowInfo.hasParent().value())
{
val
windowManager
=
this.context?.getSystemService(Context.WINDOW_SERVICE)
as
WindowManager
windowManager.addView(windowInfo.view,
windowInfo.layoutParams)
}
}
override
fun
onTouch(v:
View,
event:
MotionEvent):
Boolean
{
return
false
}
}
復(fù)制代碼public
class
GestureDetector
{
public
interface
OnGestureListener
{
//'ACTION_DOWN事件'
boolean
onDown(MotionEvent
e);
//'單擊事件'
boolean
onSingleTapUp(MotionEvent
e);
//'拖拽事件'
boolean
onScroll(MotionEvent
e1,
MotionEvent
e2,
float
distanceX,
float
distanceY);
...
}
}
復(fù)制代碼object
FloatWindow
:
View.OnTouchListener{
private
var
gestureDetector:
GestureDetector
=
GestureDetector(context,
GestureListener())
private
var
clickListener:
WindowClickListener?
=
null
private
var
lastTouchX:
Int
=
0
private
var
lastTouchY:
Int
=
0
//'為浮窗設(shè)置點(diǎn)擊監(jiān)聽器'
fun
setClickListener(listener:
WindowClickListener)
{
clickListener
=
listener
}
override
fun
onTouch(v:
View,
event:
MotionEvent):
Boolean
{
//'將觸摸事件傳遞給
GestureDetector
解析'
gestureDetector.onTouchEvent(event)
return
true
}
//'記憶起始觸摸點(diǎn)坐標(biāo)'
private
fun
onActionDown(event:
MotionEvent)
{
lastTouchX
=
event.rawX.toInt()
lastTouchY
=
event.rawY.toInt()
}
private
class
GestureListener
:
GestureDetector.OnGestureListener
{
//'記憶起始觸摸點(diǎn)坐標(biāo)'
override
fun
onDown(e:
MotionEvent):
Boolean
{
onActionDown(e)
return
false
}
override
fun
onSingleTapUp(e:
MotionEvent):
Boolean
{
//'點(diǎn)擊事件發(fā)生時(shí),調(diào)用監(jiān)聽器'
return
clickListener?.onWindowClick(windowInfo)
?:
false
}
...
}
//'浮窗點(diǎn)擊監(jiān)聽器'
interface
WindowClickListener
{
fun
onWindowClick(windowInfo:
WindowInfo?):
Boolean
}
}
復(fù)制代碼拖拽浮窗object
FloatWindow
:
View.OnTouchListener{
private
var
gestureDetector:
GestureDetector
=
GestureDetector(context,
GestureListener())
private
var
lastTouchX:
Int
=
0
private
var
lastTouchY:
Int
=
0
override
fun
onTouch(v:
View,
event:
MotionEvent):
Boolean
{
//'將觸摸事件傳遞給GestureDetector解析'
gestureDetector.onTouchEvent(event)
return
true
}
private
class
GestureListener
:
GestureDetector.OnGestureListener
{
override
fun
onDown(e:
MotionEvent):
Boolean
{
onActionDown(e)
return
false
}
override
fun
onScroll(e1:
MotionEvent,e2:
MotionEvent,distanceX:
Float,distanceY:Float):
Boolean
{
//'響應(yīng)手指滾動(dòng)事件'
onActionMove(e2)
return
true
}
}
private
fun
onActionMove(event:
MotionEvent)
{
//'獲取當(dāng)前手指坐標(biāo)'
val
currentX
=
event.rawX.toInt()
val
currentY
=
event.rawY.toInt()
//'獲取手指移動(dòng)增量'
val
dx
=
currentX
-
lastTouchX
val
dy
=
currentY
-
lastTouchY
//'將移動(dòng)增量應(yīng)用到窗口布局參數(shù)上'
windowInfo?.layoutParams!!.x
+=
dx
windowInfo?.layoutParams!!.y
+=
dy
val
windowManager
=
context?.getSystemService(Context.WINDOW_SERVICE)
as
WindowManager
var
rightMost
=
screenWidth
-
windowInfo?.layoutParams!!.width
var
leftMost
=
0
val
topMost
=
0
val
bottomMost
=
screenHeight
-
windowInfo?.layoutParams!!.height
-
getNavigationBarHeight(context)
//'將浮窗移動(dòng)區(qū)域限制在屏幕內(nèi)'
if
(windowInfo?.layoutParams!!.x
<
leftMost)
{
windowInfo?.layoutParams!!.x
=
leftMost
}
if
(windowInfo?.layoutParams!!.x
>
rightMost)
{
windowInfo?.layoutParams!!.x
=
rightMost
}
if
(windowInfo?.layoutParams!!.y
<
topMost)
{
windowInfo?.layoutParams!!.y
=
topMost
}
if
(windowInfo?.layoutParams!!.y
>
bottomMost)
{
windowInfo?.layoutParams!!.y
=
bottomMost
}
//'更新浮窗位置'
windowManager.updateViewLayout(windowInfo?.view,
windowInfo?.layoutParams)
lastTouchX
=
currentX
lastTouchY
=
currentY
}
}
復(fù)制代碼浮窗自動(dòng)貼邊object
FloatWindow
:
View.OnTouchListener{
private
var
gestureDetector:
GestureDetector
=
GestureDetector(context,
GestureListener())
private
var
lastTouchX:
Int
=
0
private
var
lastTouchY:
Int
=
0
//'貼邊動(dòng)畫'
private
var
weltAnimator:
ValueAnimator?
=
null
override
fun
onTouch(v:
View,
event:
MotionEvent):
Boolean
{
//'將觸摸事件傳遞給GestureDetector解析'
gestureDetector.onTouchEvent(event)
//'處理ACTION_UP事件'
val
action
=
event.action
when
(action)
{
MotionEvent.ACTION_UP
->
onActionUp(event,
screenWidth,
windowInfo?.width
?:
0)
else
->
{
}
}
return
true
}
private
fun
onActionUp(event:
MotionEvent,
screenWidth:
Int,
width:
Int)
{
if
(!windowInfo?.hasView().value())
{
return
}
//'記錄抬手橫坐標(biāo)'
val
upX
=
event.rawX.toInt()
//'貼邊動(dòng)畫終點(diǎn)橫坐標(biāo)'
val
endX
=
if
(upX
>
screenWidth
/
2)
{
screenWidth
-
width
}
else
{
0
}
//'構(gòu)建貼邊動(dòng)畫'
if
(weltAnimator
==
null)
{
weltAnimator
=
ValueAnimator.ofInt(windowInfo?.layoutParams!!.x,
endX).apply
{
interpolator
=
LinearInterpolator()
duration
=
300
addUpdateListener
{
animation
->
val
x
=
animation.animatedValue
as
Int
if
(windowInfo?.layoutParams
!=
null)
{
windowInfo?.layoutParams!!.x
=
x
}
val
windowManager
=
context?.getSystemService(Context.WINDOW_SERVICE)
as
WindowManager
//'更新窗口位置'
if
(windowInfo?.hasParent().value())
{
windowManager.updateViewLayout(windowInfo?.view,
windowInfo?.layoutParams)
}
}
}
}
weltAnimator?.setIntValues(windowInfo?.layoutParams!!.x,
endX)
weltAnimator?.start()
}
//為空Boolean提供默認(rèn)值
fun
Boolean?.value()
=
this
?:
false}
復(fù)制代碼管理多個(gè)浮窗object
FloatWindow
:
View.OnTouchListener
{
//'浮窗參數(shù)容器'
private
var
windowInfoMap:
HashMap<String,
WindowInfo?>
=
HashMap()
//'當(dāng)前浮窗參數(shù)'
var
windowInfo:
WindowInfo?
=
null
//'顯示浮窗'
fun
show(
context:
Context,
//'浮窗標(biāo)簽'
tag:
String,
//'若不提供浮窗參數(shù)則從參數(shù)容器中獲取該tag上次保存的參數(shù)'
windowInfo:
WindowInfo?
=
windowInfoMap[tag],
x:
Int
=
windowInfo?.layoutParams?.x.value(),
y:
Int
=
windowInfo?.layoutParams?.y.value()
)
{
if
(windowInfo
==
null)
{
return
}
if
(windowInfo.view
==
null)
{
return
}
//'更新當(dāng)前浮窗參數(shù)'
this.windowInfo
=
windowInfo
//'將浮窗參數(shù)存入容器'
windowInfoMap[tag]
=
windowInfo
windowInfo.view?.setOnTouchListener(this)
this.context
=
context
windowInfo.layoutParams
=
createLayoutParam(x,
y)
if
(!windowInfo.hasParent().value())
{
val
windowManager
=this.context?.getSystemService(Context.WINDOW_SERVICE)
as
WindowManager
windowManager.addView(windowInfo.view,
windowInfo.layoutParams)
}
}
}
復(fù)制代碼監(jiān)聽浮窗界外點(diǎn)擊事件public
class
PopupWindow
{
/**
*
<p>Controls
whether
the
pop-up
will
be
informed
of
touch
events
outside
*
of
its
window.
*
*
@param
touchable
true
if
the
popup
should
receive
outside
*
touch
events,
false
otherwise
*/
public
void
setOutsideTouchable(boolean
touchable)
{
mOutsideTouchable
=
touchable;
}
}
復(fù)制代碼public
class
PopupWindow
{
private
int
computeFlags(int
curFlags)
{
curFlags
&=
~(
WindowManager.LayoutParams.FLAG_IGNORE_CHEEK_PRESSES
|
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
|
WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE
|
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH
|
WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS
|
WindowManager.LayoutParams.FLAG_ALT_FOCUSABLE_IM
|
WindowManager.LayoutParams.FLAG_SPLIT_TOUCH);
...
//'如果界外可觸摸,則將FLAG_WATCH_OUTSIDE_TOUCH賦值給flag'
if
(mOutsideTouchable)
{
curFlags
|=
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH;
}
...
}
}
復(fù)制代碼public
class
PopupWindow
{
protected
final
WindowManager.LayoutParams
createPopupLayoutParams(IBinder
token)
{
final
WindowManager.LayoutParams
p
=
new
WindowManager.LayoutParams();
p.gravity
=
computeGravity();
//'計(jì)算窗口布局參數(shù)flag屬性并賦值'
p.flags
=
computeFlags(p.flags);
p.type
=
mWindowLayoutType;
p.token
=
token;
...
}
}
復(fù)制代碼public
class
PopupWindow
{
public
void
showAtLocation(IBinder
token,
int
gravity,
int
x,
int
y)
{
if
(isShowing()
||
mContentView
==
null)
{
return;
}
TransitionManager.endTransitions(mDecorView);
detachFromAnchor();
mIsShowing
=
true;
mIsDropdown
=
false;
mGravity
=
gravity;
//'構(gòu)建窗口布局參數(shù)'
final
WindowManager.LayoutParams
p
=
createPopupLayoutParams(token);
preparePopup(p);
p.x
=
x;
p.y
=
y;
invokePopup(p);
}
}
復(fù)制代碼public
class
PopupWindow
{
//'窗口根視圖'
private
class
PopupDecorView
extends
FrameLayout
{
//'窗口根視圖觸摸事件'
@Override
public
boolean
onTouchEvent(MotionEvent
event)
{
final
int
x
=
(int)
event.getX();
final
int
y
=
(int)
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 2025年寶雞市西府天地管委辦招考管理單位筆試遴選500模擬題附帶答案詳解
- 2025年宜昌市直事業(yè)單位招考及管理單位筆試遴選500模擬題附帶答案詳解
- 2025年安陽市紅旗渠干部學(xué)院副院長招考管理單位筆試遴選500模擬題附帶答案詳解
- 2025年安徽黃山市直事業(yè)單位招聘歷年管理單位筆試遴選500模擬題附帶答案詳解
- 2025年安徽鰲峰建設(shè)集團(tuán)限公司員工招聘(第三批次)管理單位筆試遴選500模擬題附帶答案詳解
- 2025-2030年中國無熱再生壓縮空氣干燥器項(xiàng)目可行性研究報(bào)告
- 2025-2030年中國手術(shù)床市場(chǎng)發(fā)展現(xiàn)狀規(guī)劃分析報(bào)告
- 2024-2030年脂肪乳輸液醫(yī)院用藥搬遷改造項(xiàng)目可行性研究報(bào)告
- 2024-2030年電子書閱讀器搬遷改造項(xiàng)目可行性研究報(bào)告
- 2024-2030年撰寫:中國樓宇對(duì)講系統(tǒng)項(xiàng)目風(fēng)險(xiǎn)評(píng)估報(bào)告
- 2023年環(huán)境保護(hù)部南京環(huán)境科學(xué)研究所招聘筆試參考題庫附帶答案詳解
- 繪本故事62蚯蚓的日記
- 超星爾雅學(xué)習(xí)通《西廂記》賞析(首都師范大學(xué))網(wǎng)課章節(jié)測(cè)試答案
- 新概念英語第三冊(cè)課文(全60課)
- 浙江省某住宅樓質(zhì)量通病防治措施
- YY/T 0506.1-2023醫(yī)用手術(shù)單、手術(shù)衣和潔凈服第1部分:通用要求
- TCIIA 020-2022 科學(xué)數(shù)據(jù) 安全傳輸技術(shù)要求
- GB 7101-2022食品安全國家標(biāo)準(zhǔn)飲料
- 經(jīng)濟(jì)思想史 全套講義
- 華能萊蕪電廠1000MW汽輪機(jī)圖片
- Unit 3 On the move Understanding ideas(Running into a better life)課件- 高一上學(xué)期英語外研版(2019)必修第二冊(cè)
評(píng)論
0/150
提交評(píng)論