Android WindowManager實現懸浮窗效果 (一)——與當前Activity綁定

来源:http://www.cnblogs.com/jerehedu/archive/2016/01/19/5142671.html
-Advertisement-
Play Games

最近有學生做畢業設計,想使用懸浮窗這種效果,其實很簡單,我們可以通過系統服務WindowManager來實現此功能,本章我們來試驗一下在當前Activity之上創建一個懸浮的view。第一步:認識WindowManagerl 這個介面用於與 window manager (視窗管理器, 應用框架.....


  最近有學生做畢業設計,想使用懸浮窗這種效果,其實很簡單,我們可以通過系統服務WindowManager來實現此功能,本章我們來試驗一下在當前Activity之上創建一個懸浮的view。

第一步:認識WindowManager

l  這個介面用於與 window manager (視窗管理器, 應用框架層) 進行交互。

l  通過getSystemService(Context.WINDOW_SERVICE)可以獲取到WM的實例.

l  繼承關係

       public interface WindowManager implements ViewManager

l  所屬包

       android.view.WindowManager

l  重要方法

addView()             添加view

removeView()          刪除view

updateViewLayout ()     改變view的參數

  Window Manager Service 是全局的,是唯一的。 它將用戶的操作,翻譯成為指令,發送給呈現在界面上的各個WindowActivity會將頂級的控制項註冊到 Window Manager 中,當用戶真是觸碰屏幕或鍵盤的時候,Window Manager就會通知到,而當控制項有一些請求產生,也會經由ViewParent送回到Window Manager中。從而完成整個通信流程

 

第二步:重寫ImageView 的onTouchEvent方法

  上一步我們知道了 WindowManager可以添加,刪除,改變view,那麼想要實現懸浮窗的拖動效果我們就要獲取ImageView的坐標位置。

l  獲取相對屏幕的坐標,即以屏幕左上角為原點

       float  x = event.getRawX();

       float  y = event.getRawY()-25;   //25是系統狀態欄的高度

l  通過WindowManager.LayoutParams wmParams 設置 x ,y

wmParams.x=(int)( x-mTouchStartX);

        wmParams.y=(int) (y-mTouchStartY);

l  再通過updateViewLayout()方法設置懸浮窗的當前位置

第三步:加入許可權

  在AndroidManifest.xml中加入如下的許可權:

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

效果如下:

重要代碼 :   創建 MyApplication

import android.app.Application;
import android.view.WindowManager;

public class MyApplication extends Application {

    /**
     * 創建全局變數
     * 註意在AndroidManifest.xml中的Application節點添加android:name=".MyApplication"屬性
     *
     */
    private WindowManager.LayoutParams wmParams=new WindowManager.LayoutParams();

    public WindowManager.LayoutParams getMywmParams(){
        return wmParams;
    }
}

創建自定義View 繼承ImageView

import android.content.Context;
import android.util.Log;
import android.view.MotionEvent;
import android.view.WindowManager;
import android.widget.ImageView;

public class MyFloatView extends ImageView {
    private float mTouchStartX;
    private float mTouchStartY;
    private float x;
    private float y;


    private WindowManager wm=(WindowManager)getContext().getApplicationContext().getSystemService(Context.WINDOW_SERVICE);

    //此wmParams為獲取的全局變數,用以保存懸浮視窗的屬性
    private WindowManager.LayoutParams wmParams = ((MyApplication)getContext().getApplicationContext()).getMywmParams();

    public MyFloatView(Context context) {
        super(context);
        // TODO Auto-generated constructor stub
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        //獲取相對屏幕的坐標,即以屏幕左上角為原點
        x = event.getRawX();
        y = event.getRawY()-25;   //25是系統狀態欄的高度
        Log.i("currP", "currX"+x+"====currY"+y);
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                //獲取相對View的坐標,即以此View左上角為原點
                mTouchStartX =  event.getX();
                mTouchStartY =  event.getY();
                break;
            case MotionEvent.ACTION_MOVE:
                updateViewPosition();
                break;
            case MotionEvent.ACTION_UP:
                updateViewPosition();
                mTouchStartX=mTouchStartY=0;
                break;
        }
        return true;
    }

    private void updateViewPosition(){
        //更新浮動視窗位置參數
        wmParams.x=(int)( x-mTouchStartX);
        wmParams.y=(int) (y-mTouchStartY);
        wm.updateViewLayout(this, wmParams);
    }

}

創建Activity

import android.app.Activity;

import android.content.Context;
import android.graphics.PixelFormat;
import android.os.Bundle;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.view.WindowManager;
import android.view.View.OnClickListener;
import android.view.WindowManager.LayoutParams;

public class MyFloatViewActivity extends Activity{

    private WindowManager wm=null;
    private WindowManager.LayoutParams wmParams=null;
    private MyFloatView myFV=null;
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        //創建懸浮視窗
        createView();
    }
    private void createView(){
        myFV=new MyFloatView(getApplicationContext());
        myFV.setImageResource(R.drawable.angry_birds);
        //獲取WindowManager
        wm=(WindowManager)getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
        //設置LayoutParams(全局變數)相關參數
        wmParams = ((MyApplication)getApplication()).getMywmParams();
        wmParams.type=LayoutParams.TYPE_PHONE;   //設置window type
        wmParams.format=PixelFormat.RGBA_8888;   //設置圖片格式,效果為背景透明
        //設置Window flag
        wmParams.flags=LayoutParams.FLAG_NOT_TOUCH_MODAL
                | LayoutParams.FLAG_NOT_FOCUSABLE;
        wmParams.gravity=Gravity.LEFT|Gravity.TOP;   //調整懸浮視窗至左上角
        //以屏幕左上角為原點,設置x、y初始值
        wmParams.x=0;
        wmParams.y=0;
        //設置懸浮視窗長寬數據
        wmParams.width=40;
        wmParams.height=40;
        //顯示myFloatView圖像
        wm.addView(myFV, wmParams);

    }
    @Override
    public void onDestroy(){
        super.onDestroy();
        //在程式退出(Activity銷毀)時銷毀懸浮視窗
        wm.removeView(myFV);
    }
}

l  最後在程式安裝時修改手機里的程式許可權-》懸浮窗可用

 

作者:傑瑞教育
出處:http://www.cnblogs.com/jerehedu/ 
版權聲明:本文版權歸傑瑞教育技有限公司和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。
技術咨詢:JRedu技術交流
 
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Swift開發體驗/*:創建對象* OC: alloc initWithXXX 方法* Swift: (xxx:)*//*:調用方法* OC: [UIColor redColor];* Swift UIColor.redColor()*//*:枚舉* OC: UIB...
  • 一、weak和strong 1.理解 剛開始學UI的時候,對於weak和strong的描述看得最多的就是“由ARC引入,weak相當於OC中的assign,但是weak用於修飾對象,但是他們都不會造成引用計數加1;而strong則相當於OC中規定retain,它會造成引用計數加1”。 ARC的...
  • 背景最近要做一個輪播圖的效果,網上看了幾篇文章,基本上都能找到實現,效果還挺不錯,但是在寫的時候感覺每次都要單獨去重新在Activity里寫一堆代碼。於是自己封裝了一下。本篇輪播圖實現原理原文出處:迴圈廣告位組件的實現,這裡只是做了下封裝成一個控制項,不必每次重覆寫代碼了。效果圖實現分析輪播圖的功能就...
  • NSAttributedString的使用 使label具有點擊事件 定義帶參數巨集的註意事項 pch文件基本使用規則 使用CocoaPods集成Masonry框架 Xcode7中VVDocument插件解決方案
  • 參考:http://www.cnblogs.com/hubli/p/4835549.html效果圖:1.wevbview_progressbar.xml ...
  • 主要用於UITableView和UICollectionView,也可以用於UIScrollView,其實主要是前兩個會用到空白或者網路出錯頁 採用給UIScrollView添加代理方法來給頁面添加空白頁,源碼很有學習意義
  • 文章內容大綱1、NSMutableAttributedString的基本使用2、NSMutableAttributedString的簡易封裝3、使用開源代碼GOBMarkupPaser處理富文本4、UITextKit簡介5、編程思想的相關思考前言富文本使用案例:這裡我自己也用了富文本實現了簡單的卻也...
  • 一、沙盒(SandBox) 1.沙盒機制 1> 每個應用都有屬於自己的存儲空間,即沙盒。 2> 應用只能訪問自己的沙盒,不可訪問其他區域。 3> 如果應用需要進行文件操作,則必須將文件存放在沙盒中,尤其是資料庫文件,在電腦上操作時,可以去訪問,但是如果要裝在真機上可以使用,必須將資料庫文件...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...