私は常にすべてのウィンドウの一番上にとどまるように常に操作可能なトップボタン/クリック可能イメージを作成しようとしています。
概念実証は
私は成功しており、現在実行中のサービスを持っています。ユーザーが通常の方法で他のアプリと自由にやり取りできるようにしながら、サービスは常に画面の左上隅にいくつかのテキストを表示します。
私がやっているのはサブクラスViewGroup
で、それをフラグTYPE_SYSTEM_OVERLAY
を付けてrootウィンドウマネージャに追加します。このテキストの代わりに、それ自体でタッチイベントを受け取ることができるボタン/クリック可能な画像を追加したいと思います。私は全体のViewGroup
に対して "onTouchEvent"を上書きしようとしましたが、イベントを受け取りません。
トップビューグループの特定の部分でのみイベントを受信するにはどうすればよいですか。親切にお勧めします。
public class HUD extends Service {
HUDView mView;
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onCreate() {
super.onCreate();
Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();
mView = new HUDView(this);
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
0,
// WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
// | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE,
PixelFormat.TRANSLUCENT);
params.gravity = Gravity.RIGHT | Gravity.TOP;
params.setTitle("Load Average");
WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
wm.addView(mView, params);
}
@Override
public void onDestroy() {
super.onDestroy();
Toast.makeText(getBaseContext(),"onDestroy", Toast.LENGTH_LONG).show();
if(mView != null)
{
((WindowManager) getSystemService(WINDOW_SERVICE)).removeView(mView);
mView = null;
}
}
}
class HUDView extends ViewGroup {
private Paint mLoadPaint;
public HUDView(Context context) {
super(context);
Toast.makeText(getContext(),"HUDView", Toast.LENGTH_LONG).show();
mLoadPaint = new Paint();
mLoadPaint.setAntiAlias(true);
mLoadPaint.setTextSize(10);
mLoadPaint.setARGB(255, 255, 0, 0);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawText("Hello World", 5, 15, mLoadPaint);
}
@Override
protected void onLayout(boolean arg0, int arg1, int arg2, int arg3, int arg4) {
}
@Override
public boolean onTouchEvent(MotionEvent event) {
//return super.onTouchEvent(event);
Toast.makeText(getContext(),"onTouchEvent", Toast.LENGTH_LONG).show();
return true;
}
}
これは愚かな解決策かもしれません。しかしそれはうまくいきます。あなたがそれを改善することができるならば、私に知らせてください。
あなたのサービスの作成:私はWindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH
フラグを使いました。これがサービスの唯一の変更点です。
@Override
public void onCreate() {
super.onCreate();
Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();
mView = new HUDView(this);
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSLUCENT);
params.gravity = Gravity.RIGHT | Gravity.TOP;
params.setTitle("Load Average");
WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
wm.addView(mView, params);
}
今、あなたは一人ひとりクリックイベントを取得し始めます。だから、あなたはあなたのイベントハンドラで修正する必要があります。
ViewGroupタッチイベントで
@Override
public boolean onTouchEvent(MotionEvent event) {
// ATTENTION: GET THE X,Y OF EVENT FROM THE PARAMETER
// THEN CHECK IF THAT IS INSIDE YOUR DESIRED AREA
Toast.makeText(getContext(),"onTouchEvent", Toast.LENGTH_LONG).show();
return true;
}
また、あなたはあなたの目録にこの許可を加える必要があるかもしれません:
<uses-permission Android:name="Android.permission.SYSTEM_ALERT_WINDOW" />
@Sam Luの回答を受けて、確かにAndroid 4.xは特定のタイプが外部のタッチイベントをリスニングするのをブロックしますが、TYPE_SYSTEM_ALERT
のようないくつかのタイプはまだ仕事をします。
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
WindowManager.LayoutParams.MATCH_PARENT,
WindowManager.LayoutParams.MATCH_PARENT,
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
| WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
| WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSLUCENT);
WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);
View myView = inflater.inflate(R.layout.my_view, null);
myView.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
Log.d(TAG, "touch me");
return true;
}
});
// Add layout to window manager
wm.addView(myView, params);
<uses-permission Android:name="Android.permission.SYSTEM_ALERT_WINDOW"/>
Android 4.x以降、Androidチームは潜在的なセキュリティ問題を解決し、FLAG_NOT_FOCUSABLE
ウィンドウのFLAG_NOT_TOUCHABLE
、FLAG_WATCH_OUTSIDE_TOUCH
を追加し、TYPE_SYSTEM_OVERLAY
フラグを削除する新しい関数adjustWindowParamsLw()
を追加しました。
つまり、TYPE_SYSTEM_OVERLAY
ウィンドウはICSプラットフォームではタッチイベントを受け取らず、TYPE_SYSTEM_OVERLAY
を使用することは_ICSおよび将来のデバイスにとって実用的な解決策ではありません。
私は Tooleap SDK の開発者の一人です。開発者が常に一番上のウィンドウやボタンに表示する方法も提供していますが、同様の状況に対処しました。
ここでの答えが解決していない問題の1つは、Androidの「保護されたボタン」の問題です。
保護されたボタンはfilterTouchesWhenObscured
プロパティを持っています。これは、ウィンドウの下に配置されている場合、そのウィンドウがタッチを受けていなくても操作できないことを意味します。 Androidのドキュメントを引用する:
ビューのウィンドウが別の表示ウィンドウによって隠されているときにタッチをフィルタ処理するかどうかを指定します。 trueに設定すると、トースト、ダイアログ、またはその他のウィンドウがビューのウィンドウの上に表示されるたびに、ビューはタッチを受け取りません。詳細については{@link Android.view.View}セキュリティドキュメントを参照してください。
このようなボタンの例としては、サードパーティ製APKをインストールしようとしたときのインストールボタンがあります。ビューレイアウトに次の行を追加すると、どのアプリでもこのようなボタンを表示できます。
Android:filterTouchesWhenObscured="true"
「セキュリティで保護されたボタン」の上に常に最前面のウィンドウを表示すると、そのオーバーレイがクリックできなくても、オーバーレイで覆われているすべてのセキュリティ保護されたボタン部分はタッチを処理できません。したがって、そのようなウィンドウを表示することを計画している場合は、ユーザーがそれを移動するかそれを閉じるための方法を提供する必要があります。また、オーバーレイの一部が透明になっている場合は、基礎となるアプリの特定のボタンが突然機能しなくなっているのはユーザーが混乱している可能性があることを考慮してください。
上の画像ボタンで常に作業する
まず第一に私の英語をごめんなさい
私はあなたのコードを編集し、彼のタッチイベントを聞く作業イメージボタンを彼の背景の要素にタッチコントロールを与えないようにします。
また、それは他の要素からタッチリスナーに与える
ボタンの配置は下と左
あなたはalingmentsを変更することができますが、あなたはif要素のtouchイベントで座標を変更する必要があります
import Android.annotation.SuppressLint;
import Android.app.Service;
import Android.content.Context;
import Android.content.Intent;
import Android.graphics.Bitmap;
import Android.graphics.BitmapFactory;
import Android.graphics.Canvas;
import Android.graphics.Color;
import Android.graphics.PixelFormat;
import Android.os.IBinder;
import Android.util.Log;
import Android.view.Gravity;
import Android.view.MotionEvent;
import Android.view.View;
import Android.view.View.OnTouchListener;
import Android.view.ViewGroup;
import Android.view.WindowManager;
import Android.widget.Toast;
public class HepUstte extends Service {
HUDView mView;
@Override
public void onCreate() {
super.onCreate();
Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();
final Bitmap kangoo = BitmapFactory.decodeResource(getResources(),
R.drawable.logo_l);
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
kangoo.getWidth(),
kangoo.getHeight(),
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
|WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
|WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSLUCENT);
params.gravity = Gravity.LEFT | Gravity.BOTTOM;
params.setTitle("Load Average");
WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
mView = new HUDView(this,kangoo);
mView.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View arg0, MotionEvent arg1) {
// TODO Auto-generated method stub
//Log.e("kordinatlar", arg1.getX()+":"+arg1.getY()+":"+display.getHeight()+":"+kangoo.getHeight());
if(arg1.getX()<kangoo.getWidth() & arg1.getY()>0)
{
Log.d("tıklandı", "touch me");
}
return false;
}
});
wm.addView(mView, params);
}
@Override
public IBinder onBind(Intent arg0) {
// TODO Auto-generated method stub
return null;
}
}
@SuppressLint("DrawAllocation")
class HUDView extends ViewGroup {
Bitmap kangoo;
public HUDView(Context context,Bitmap kangoo) {
super(context);
this.kangoo=kangoo;
}
protected void onDraw(Canvas canvas) {
//super.onDraw(canvas);
// delete below line if you want transparent back color, but to understand the sizes use back color
canvas.drawColor(Color.BLACK);
canvas.drawBitmap(kangoo,0 , 0, null);
//canvas.drawText("Hello World", 5, 15, mLoadPaint);
}
protected void onLayout(boolean arg0, int arg1, int arg2, int arg3, int arg4) {
}
public boolean onTouchEvent(MotionEvent event) {
//return super.onTouchEvent(event);
// Toast.makeText(getContext(),"onTouchEvent", Toast.LENGTH_LONG).show();
return true;
}
}
実際には、TYPE_SYSTEM_OVERLAYの代わりにWindowManager.LayoutParams.TYPE_SYSTEM_ERRORを試すことができます。それはハックのように聞こえるかもしれませんが、それはあなたがすべての上にビューを表示してもタッチイベントを取得することを可能にします。
これを試して。 ICSで問題なく動作します。あなたがサービスを停止したい場合は、単にステータスバーに生成された通知をクリックしてください。
public class HUD extends Service
{
protected boolean foreground = false;
protected boolean cancelNotification = false;
private Notification notification;
private View myView;
protected int id = 0;
private WindowManager wm;
private WindowManager.LayoutParams params;
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onCreate() {
super.onCreate();
// System.exit(0);
Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_SHORT).show();
params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT, WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.TYPE_PHONE, WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
| WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, PixelFormat.TRANSLUCENT);
params.gravity=Gravity.TOP|Gravity.LEFT;
wm = (WindowManager) getSystemService(WINDOW_SERVICE);
inflateview();
foregroundNotification(1);
//moveToForeground(1,n,true);
}
@Override
public void onDestroy() {
super.onDestroy();
((NotificationManager) getSystemService(NOTIFICATION_SERVICE)).cancel(0);
Toast.makeText(getBaseContext(),"onDestroy", Toast.LENGTH_SHORT).show();
if(myView != null)
{
((WindowManager) getSystemService(WINDOW_SERVICE)).removeView(myView);
myView = null;
}
}
protected Notification foregroundNotification(int notificationId)
{
notification = new Notification(R.drawable.ic_launcher, "my Notification", System.currentTimeMillis());
notification.flags = notification.flags | Notification.FLAG_ONGOING_EVENT | Notification.FLAG_ONLY_ALERT_ONCE;
notification.setLatestEventInfo(this, "my Notification", "my Notification", notificationIntent());
((NotificationManager) getSystemService(NOTIFICATION_SERVICE)).notify(id, notification);
return notification;
}
private PendingIntent notificationIntent() {
Intent intent = new Intent(this, stopservice.class);
PendingIntent pending = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
return pending;
}
public void inflateview()
{
LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);
myView = inflater.inflate(R.layout.activity_button, null);
myView.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
Toast.makeText(getBaseContext(),"onToasttt", Toast.LENGTH_SHORT).show();
return false;
}
});
// Add layout to window manager
wm.addView(myView, params);
}
}
UPDATE
サンプル こちら
オーバーレイビューを作成するには、LayoutParamsを設定するときにタイプをTYPE_SYSTEM_OVERLAYに設定しないでください。
Instead set it to TYPE_PHONE.
Use the following flags:
FLAG_NOT_TOUCH_MODAL
FLAG_WATCH_OUTSIDE_TOUCH
TYPE_SYSTEM_OVERLAYこの定数は、APIレベル26以降では使用されなくなりました。代わりにTYPE_APPLICATION_OVERLAYを使用してください。
それはこのリンク上でパーミッション "Android.permission.SYSTEM_ALERT_WINDOW"を使用しています。 http://androidsrc.net/facebook-chat-like-floating-chat-heads/
ここにいくつかの簡単な解決策があります、あなたが必要とするすべてはあなたがそれを膨らませるためにXMLレイアウトを作るようにちょうどリストアダプタでするようにXMLレイアウトを膨らませることです。これがあなたが必要とする全てのコードです。
public class HUD extends Service {
View mView;
LayoutInflater inflate;
TextView t;
Button b;
@Override
public void onCreate() {
super.onCreate();
Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();
WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
Display display = wm.getDefaultDisplay(); get phone display size
int width = display.getWidth(); // deprecated - get phone display width
int height = display.getHeight(); // deprecated - get phone display height
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
width,
height,
WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
|WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
|WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
PixelFormat.TRANSLUCENT);
params.gravity = Gravity.LEFT | Gravity.CENTER;
params.setTitle("Load Average");
inflate = (LayoutInflater) getBaseContext()
.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
mView = inflate.inflate(R.layout.canvas, null);
b = (Button) mView.findViewById(R.id.button1);
t = (TextView) mView.findViewById(R.id.textView1);
b.setOnClickListener(new OnClickListener() {
public void onClick(View v) {
// TODO Auto-generated method stub
t.setText("yes you click me ");
}
});
wm.addView(mView, params);
}
@Override
public IBinder onBind(Intent arg0) {
// TODO Auto-generated method stub
return null;
}
}
ちょうどそれをするライブラリを見つけました: https://github.com/recruit-lifestyle/FloatingView
ルートフォルダにサンプルプロジェクトがあります。私はそれを実行しました、そしてそれは要求通りに働きます。他のアプリであっても、背景はクリック可能です。
サービスを使用することでこれを実現できます。
public class PopupService extends Service{
private static final String TAG = PopupService.class.getSimpleName();
WindowManager mWindowManager;
View mView;
String type ;
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
// registerOverlayReceiver();
type = intent.getStringExtra("type");
Utils.printLog("type = "+type);
showDialog(intent.getStringExtra("msg"));
return super.onStartCommand(intent, flags, startId);
}
private void showDialog(String aTitle)
{
if(type.equals("when screen is off") | type.equals("always"))
{
Utils.printLog("type = "+type);
PowerManager pm = (PowerManager) getApplicationContext().getSystemService(Context.POWER_SERVICE);
WakeLock mWakeLock = pm.newWakeLock((PowerManager.SCREEN_DIM_WAKE_LOCK | PowerManager.ACQUIRE_CAUSES_WAKEUP), "YourServie");
mWakeLock.acquire();
mWakeLock.release();
}
mWindowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
mView = View.inflate(getApplicationContext(), R.layout.dialog_popup_notification_received, null);
mView.setTag(TAG);
int top = getApplicationContext().getResources().getDisplayMetrics().heightPixels / 2;
LinearLayout dialog = (LinearLayout) mView.findViewById(R.id.pop_exit);
// Android.widget.LinearLayout.LayoutParams lp = (Android.widget.LinearLayout.LayoutParams) dialog.getLayoutParams();
// lp.topMargin = top;
// lp.bottomMargin = top;
// mView.setLayoutParams(lp);
final EditText etMassage = (EditText) mView.findViewById(R.id.editTextInPopupMessageReceived);
ImageButton imageButtonSend = (ImageButton) mView.findViewById(R.id.imageButtonSendInPopupMessageReceived);
// lp = (LayoutParams) imageButton.getLayoutParams();
// lp.topMargin = top - 58;
// imageButton.setLayoutParams(lp);
imageButtonSend.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Utils.printLog("clicked");
// mView.setVisibility(View.INVISIBLE);
if(!etMassage.getText().toString().equals(""))
{
Utils.printLog("sent");
etMassage.setText("");
}
}
});
TextView close = (TextView) mView.findViewById(R.id.TextViewCloseInPopupMessageReceived);
close.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
hideDialog();
}
});
TextView view = (TextView) mView.findViewById(R.id.textviewViewInPopupMessageReceived);
view.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View arg0) {
hideDialog();
}
});
TextView message = (TextView) mView.findViewById(R.id.TextViewMessageInPopupMessageReceived);
message.setText(aTitle);
final WindowManager.LayoutParams mLayoutParams = new WindowManager.LayoutParams(
ViewGroup.LayoutParams.MATCH_PARENT,
ViewGroup.LayoutParams.MATCH_PARENT, 0, 0,
WindowManager.LayoutParams.TYPE_SYSTEM_ERROR,
WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED
| WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD
// | WindowManager.LayoutParams.FLAG_TURN_SCREEN_ON
| WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON ,
PixelFormat.RGBA_8888);
mView.setVisibility(View.VISIBLE);
mWindowManager.addView(mView, mLayoutParams);
mWindowManager.updateViewLayout(mView, mLayoutParams);
}
private void hideDialog(){
if(mView != null && mWindowManager != null){
mWindowManager.removeView(mView);
mView = null;
}
}
}
私のコードを試してみてください。少なくともそれがオーバーレイとしてあなたに文字列を与えるなら、あなたはそれをボタンまたはイメージで非常によく置き換えることができます。あなたはこれが私の史上初のAndroidアプリLOLだとは思わない。とにかくあなたが私よりもAndroidアプリの経験があるなら、試してください
まだこのスレッドを読んでいて、これを動かすことができない人がいるなら、モーションイベントを傍受するためのこの方法を教えて申し訳ありませんが、Android 4.2ではバグと修正と見なされています。
あなたが傍受したモーションイベントはACTION_OUTSIDEとしてのアクションを持ちますが、getXとgetYで0を返します。これはあなたがスクリーン上の全ての動きの位置を見ることができないことを意味します、またあなたは何もすることができません。私はその文書がxとyになると言っているのを知っていますが、真実はそうではありません。これはキーロガーをブロックするためのようです。
誰かが回避策を持っている場合は、あなたのコメントを残してください。
AndroidがWindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCHでさえタッチできないようにするために、WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAYでビューをウィンドウに追加することは、Androidでは許可されていないため、@Sarwar Erfanの回答は機能しません。
私はこの問題に対する解決策を見つけました。次の質問でそれをチェックアウトすることができます
WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAYを使用してウィンドウにビューを追加しても、タッチイベントが発生していません