Android 悬浮并可拖动的实现

    http://www.2cto.com/kf/201205/132308.html

     通过下面几句代码,就可以让一个View凌驾在所有View之上。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
WindowManager wm = (WindowManager)getApplicationContext().getSystemService(WINDOW_SERVICE);  
    
WindowManager.LayoutParams params = new WindowManager.LayoutParams();  
    
params.type = WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY;  
    
     
    
params.width = WindowManager.LayoutParams.WRAP_CONTENT;  
    
params.height = WindowManager.LayoutParams.WRAP_CONTENT;  
    
     
    
TextView tv = new TextView( this );  
    
wm.addView(tv, params);


这边需要注意的是, WindowManager也是通过 getSystemService 来获取,但必须先 getApplicationContext, 否则就无效了。 
直接WindowManager wm = (WindowManager)getSystemService(WINDOW_SERVICE); 这样是无效的 !! 还有一点就是,别忘了在Manifest.xml中添加权限:

1
<uses-permission android:name= "android.permission.SYSTEM_ALERT_WINDOW" />

现在我们这样做,我们已经可以让歌词永远置顶了。 但是不要得意,现在这样,结果是我们TextView在最顶层了, 然后你就会发现,页面上什么操作都不能做了, 在TextView下面的任何东西,你都点不了。 
为了解决这个,我们必须加上flags参数,让当前的View失去焦点,从而让后面的页面获得焦点。代码如下:

1
params.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL | LayoutParams.FLAG_NOT_FOCUSABLE;

加上这一句就可以了。 
好了,下面要处理的,就是让歌词可以移动。应该如何做呢? 我们知道,想要让一个View对象在页面上可以移动,只要实现其onTouchEvent事件即可。 
  下面开始实现第二步: 歌词移动! 
首先我们自定义一个TextView类:MyTextView, 该类继承自TextView, 并实现其中的onTouchEvent方法,来看一下代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
@Override  
    
public boolean onTouchEvent(MotionEvent event) {  
    
         //触摸点相对于屏幕左上角坐标  
    
         x = event.getRawX();     
    
     y = event.getRawY() - TOOL_BAR_HIGH;  
    
     Log.d(TAG, "------X: " + x + "------Y:" + y);  
    
         
    
     switch (event.getAction()) {  
    
             case MotionEvent.ACTION_DOWN:  
    
                     startX = event.getX();  
    
                     startY = event.getY();  
    
                     break ;  
    
             case MotionEvent.ACTION_MOVE:  
    
                     updatePosition();  
    
                     break ;  
    
             case MotionEvent.ACTION_UP:  
    
                     updatePosition();  
    
                     startX = startY = 0;  
    
                     break ;  
    
     }  
    
         
    
         return true ;  
    
}  
    
//更新浮动窗口位置参数  
    
private void updatePosition(){  
    
          // View的当前位置  
    
          params.x = (int)( x - startX);  
    
          params.y = (int) (y - startY);  
    
          wm.updateViewLayout( this , params);  
    
}

其中getRawX、getRawY用于获取触摸点离屏幕左上角的距离。 而getX、getY用于获取触摸点离textView左上角的距离.两者相减,就是View左上角的坐标了。另外需要注意的是,在显示View这个View的时候,需要正确指定View的x,y坐标,否则拖动时会错位。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
WindowManager wm = (WindowManager)getApplicationContext().getSystemService(WINDOW_SERVICE);  
    
                 WindowManager.LayoutParams params = MyTextView.params;  
    
                     
    
                 params.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT | WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY;  
    
                 params.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL | LayoutParams.FLAG_NOT_FOCUSABLE;  
    
                     
    
                 params.width = WindowManager.LayoutParams.FILL_PARENT;  
    
                 params.height = WindowManager.LayoutParams.WRAP_CONTENT;  
    
                 params.alpha = 80;  
    
                     
    
                 params.gravity=Gravity.LEFT|Gravity.TOP;  
    
             //以屏幕左上角为原点,设置x、y初始值  
    
                 params.x = 0;  
    
                 params.y = 0;  
    
                     
    
                 tv = new MyTextView(TopFrame. this );  
    
                 wm.addView(tv, params);

其中下面三句是关键:

1
2
3
4
5
6
7
params.gravity=Gravity.LEFT|Gravity.TOP;  
    
幕左上角为原点,设置x、y初始值  
    
params.x = 0;  
    
params.y = 0;

现在这样的话,就可以实现View的移动了。 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值