การสร้างหน้าต่างซ้อนทับระบบ (อยู่ด้านบนเสมอ)


285

ฉันพยายามสร้างปุ่มแบบเปิดตลอดเวลา / ภาพที่คลิกได้ซึ่งอยู่ด้านบนของหน้าต่างทั้งหมดตลอดเวลา

หลักฐานของแนวคิดคือ

ฉันประสบความสำเร็จและมีบริการที่ทำงานอยู่ในขณะนี้ บริการแสดงข้อความที่มุมบนซ้ายของหน้าจอตลอดเวลาในขณะที่ผู้ใช้สามารถโต้ตอบกับแอพที่เหลือในลักษณะปกติได้อย่างอิสระ

สิ่งที่ฉันทำคือประเภทรองและเพิ่มเข้าไปจัดการหน้าต่างรากที่มีธงViewGroup TYPE_SYSTEM_OVERLAYตอนนี้ฉันต้องการเพิ่มปุ่ม / คลิกได้ - ภาพแทนข้อความนี้ซึ่งสามารถรับเหตุการณ์สัมผัสด้วยตัวเอง ฉันพยายามเอาชนะ "onTouchEvent" ทั้งหมดViewGroupแต่ไม่ได้รับเหตุการณ์ใด ๆ

ฉันจะรับกิจกรรมในบางส่วนของกลุ่มมุมมองด้านบนเสมอได้อย่างไร กรุณาแนะนำ

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;
    }
}

2
@coreSOLO ฉันหวังว่าคุณจะได้รับคำตอบคำถามที่คล้ายกันของฉันไม่ได้รับคำตอบใด ๆ :) +1
st0le

3
@ st0le ฉันสัญญาฉันจะใช้เวลาทั้งวันทั้งคืนจนกว่าฉันจะหาวิธีแก้ปัญหาและแบ่งปันกับคุณ :)
coreSOLO

1
@ st0le ตกลงบางทีคุณและฉันสามารถขุดมันออกมาด้วยกันให้ฉันเชื่อมโยงคำถามของคุณและแจ้งให้เราทราบว่าคุณสามารถสร้างความก้าวหน้าได้มากแค่ไหน ...
coreSOLO

มันไม่มาก: - \ stackoverflow.com/questions/3732935/…
st0le

3
มันน่าประหลาดใจที่คุณonDraw()ถูกเรียกตัว ไม่ควร dispatchDraw()ถูกแทนที่แทน? ดู groups.google.com/forum/?fromgroups#!topic/android-developers/?hl=th
faizal

คำตอบ:


158

นี่อาจเป็นวิธีที่โง่ แต่มันได้ผล หากคุณสามารถปรับปรุงได้โปรดแจ้งให้เราทราบ

เมื่อสร้างบริการของคุณ: ฉันได้ใช้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);
    }

ตอนนี้คุณจะเริ่มได้รับการคลิกทุกครั้ง ดังนั้นคุณต้องแก้ไขในตัวจัดการเหตุการณ์ของคุณ

ในเหตุการณ์ TouchGroup touch ของคุณ

@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" />

1
@pengwang: คุณได้รับข้อผิดพลาดอะไร? รหัสนี้ไม่เป็นอิสระคุณต้องวางไว้ในโครงการของคุณด้วยชื่อไฟล์ที่เหมาะสมภายใต้โครงการที่เหมาะสม
Sarwar Erfan

4
@pengwang: ค้นหาการดำเนินงานที่ง่ายที่สุดที่นี่: docs.google.com/...
โฉบ erfan

6
รหัสทั้งหมดถูกต้องฉันพบข้อผิดพลาดของฉันฉันลืมลิงค์ <การใช้สิทธิ์ Android: ชื่อ = "android.permission.SYSTEM_ALERT_WINDOW" /> Sarwar Erfan ฉันคิดว่าคุณแก้ไขคำตอบของคุณมันจะกลายเป็นที่ยอดเยี่ยมมากขึ้น
pengwang

1
คำตอบนี้มีประโยชน์มากยกเว้นเพียงสิ่งเดียว ตามที่ใช้ในรหัส WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH ตามเอกสาร "จะไม่ได้รับท่าทางเต็ม / เลื่อน / ลง" กล่าวอีกนัยหนึ่งไม่สามารถตรวจจับ swipes หรือสิ่งอื่นใดนอกจาก ACTION_DOWN ฉันรู้ว่ามันเป็นไปได้ที่จะตรวจจับ swipes เพราะ Wave Launcher ใช้แนวคิดที่คล้ายกันนี้และพวกเขาสามารถตรวจจับ swipes ได้ ทุกคนสามารถอธิบายให้ฉันทำงานนี้ได้อย่างไร
Brian

6
@SarwarErfan - ฉันใช้มัน corking อย่างสมบูรณ์แบบใน 2.3 แต่เหตุการณ์ onTouch ไม่ได้ geeting invocked บน Android 4.0, 1 และ 4.1 โปรดช่วย
Sanjay Singh

66

ปฏิบัติตามคำตอบของ @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"/>

5
มันทำให้หน้าจอของฉันถูกแขวนคอ .. ความคิดใด ๆ
Viswanath Lekshmanan

โพสต์ยอดเยี่ยม หวังว่านี่จะไม่ปิดการใช้งานในอนาคต
Cookster

แอนิเมชั่นจะใช้งานไม่ได้ ... :( และมันไม่สามารถจับคลิกได้ในทุกมุมมองใน 4.2.X. คำแนะนำใด ๆ ?
Kamran Ahmed

ใช้งานได้ แต่เมื่อฉันใช้ทางลัดนั้น (เช่นการแจ้งเตือนระดับระบบ) กล่องโต้ตอบอื่น ๆ จะไม่ปรากฏขึ้น ทำไม
Chimu

@Arju คุณหมายถึงอะไร คุณพยายามที่จะลบมัน?
amirlazarovich

18

เริ่มต้นกับ Android 4.x ทีม Android การแก้ไขปัญหาด้านความปลอดภัยที่อาจเกิดขึ้นโดยการเพิ่มฟังก์ชั่นใหม่adjustWindowParamsLw()ที่จะเพิ่มFLAG_NOT_FOCUSABLE, FLAG_NOT_TOUCHABLEและลบFLAG_WATCH_OUTSIDE_TOUCHธงสำหรับTYPE_SYSTEM_OVERLAYหน้าต่าง

กล่าวคือTYPE_SYSTEM_OVERLAYหน้าต่างจะไม่ได้รับเหตุการณ์การสัมผัสใด ๆ บนแพลตฟอร์ม ICS และแน่นอนว่าการใช้TYPE_SYSTEM_OVERLAYไม่ใช่โซลูชันที่ใช้งานได้สำหรับ ICS และอุปกรณ์ในอนาคต


4
ดังนั้นทางเลือกคืออะไร?
radhoo

1
ดูคำถามเกี่ยวกับ ICS นี้: stackoverflow.com/questions/9656185/type-system-overlay-in-ics
clemp6r

13

ฉันเป็นหนึ่งในนักพัฒนาของTooleap SDKSDK นอกจากนี้เรายังให้วิธีการสำหรับนักพัฒนาซอฟต์แวร์ที่จะแสดงบนหน้าต่างและปุ่มด้านบนเสมอและและเราได้จัดการกับสถานการณ์ที่คล้ายกัน

ปัญหาหนึ่งคำตอบที่นี่ยังไม่ได้แก้ไขก็คือปุ่ม "Secured Buttons" ของ Android

ปุ่มที่ปลอดภัยมีfilterTouchesWhenObscuredคุณสมบัติซึ่งหมายความว่าพวกเขาไม่สามารถโต้ตอบกับถ้าวางไว้ใต้หน้าต่างแม้ว่าหน้าต่างนั้นจะไม่ได้รับสัมผัสใด ๆ การอ้างอิงเอกสาร Android:

ระบุว่าจะให้ตัวกรองสัมผัสเมื่อหน้าต่างของมุมมองถูกบดบังโดยหน้าต่างที่มองเห็นอื่น เมื่อตั้งค่าเป็นจริงมุมมองจะไม่ได้รับการสัมผัสเมื่อใดก็ตามที่ขนมปังปิ้งกล่องโต้ตอบหรือหน้าต่างอื่นปรากฏขึ้นเหนือหน้าต่างของมุมมอง อ้างถึงเอกสารความปลอดภัย {@link android.view.View} สำหรับรายละเอียดเพิ่มเติม

ตัวอย่างของปุ่มดังกล่าวคือปุ่มติดตั้งเมื่อคุณพยายามติดตั้ง apks ของบุคคลที่สาม แอปใด ๆ สามารถแสดงปุ่มดังกล่าวหากเพิ่มไปยังเค้าโครงมุมมองบรรทัดต่อไปนี้:

android:filterTouchesWhenObscured="true"

หากคุณแสดงหน้าต่างที่อยู่ด้านบนตลอดเวลาบน "ปุ่มที่มีความปลอดภัย" ดังนั้นชิ้นส่วนของปุ่มที่มีการรักษาความปลอดภัยที่ครอบคลุมด้วยการซ้อนทับจะไม่จัดการกับการสัมผัสใด ๆ แม้ว่าการซ้อนทับนั้นจะไม่สามารถคลิกได้ ดังนั้นหากคุณวางแผนที่จะแสดงหน้าต่างดังกล่าวคุณควรจัดให้มีวิธีสำหรับผู้ใช้ในการย้ายหรือยกเลิกมัน และหากส่วนหนึ่งของการซ้อนทับของคุณโปร่งใสพิจารณาว่าผู้ใช้ของคุณอาจสับสนว่าทำไมปุ่มบางปุ่มในแอปที่ใช้งานไม่ได้ทำงานให้เขาทันที


คุณช่วยตอบคำถามของฉันได้ที่: stackoverflow.com/questions/28964771/…
Nauman Afzaal

มีบางกิจกรรมใน Android 10 ที่มุมมองภาพซ้อนทับไม่สามารถแสดงด้านบนได้โปรดบอกเหตุผลหรือดูคำถามของฉันที่นี่มันจะมีประโยชน์มากstackoverflow.com/questions/59598264/ …
Mateen Chaudhry

12

การทำงานบนปุ่มภาพด้านบนเสมอ

ก่อนอื่นต้องขออภัยในภาษาอังกฤษของฉัน

ฉันแก้ไขรหัสของคุณและทำให้ปุ่มรูปภาพใช้งานที่ฟังเหตุการณ์การสัมผัสของเขาไม่ได้ให้การควบคุมแบบสัมผัสกับองค์ประกอบพื้นหลังของเขา

นอกจากนี้ยังช่วยให้ผู้ฟังระบบสัมผัสเข้าถึงองค์ประกอบอื่น ๆ

การแจ้งเตือนปุ่มอยู่ด้านล่างและซ้าย

คุณสามารถเปลี่ยนการแจ้งเตือนได้ แต่คุณต้องเปลี่ยนลูกน้องในเหตุการณ์แบบสัมผัสในองค์ประกอบ if

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;
    }
}

ใช้งานได้ แต่ฉันจะแน่ใจได้อย่างไรว่าภาพของฉันถูกคลิกและไม่ใช่สิ่งรอบตัว?
เลียม

@LiamW พิกัดของการสัมผัสถูกเปรียบเทียบกับ kangaroo.getWidth () ในตัวอย่างนี้ คุณสามารถอนุมานสิ่งที่จะเปรียบเทียบกับในโค้ดของคุณเองตามที่คุณวาดภาพของคุณ
Yevgeny Simkin

ดีมาก! ทำงานได้อย่างสมบูรณ์แบบ
Yevgeny Simkin

Android เวอร์ชันขั้นต่ำนี้ใช้งานอะไรได้บ้าง? ฉันกำลังทำงานเพื่อเขียนสิ่งนี้ แต่ฉันใหม่กับ Android เพื่อให้มีเวลาเริ่มต้น
Noitidart

6

ที่จริงแล้วคุณสามารถลอง WindowManager.LayoutParams.TYPE_SYSTEM_ERROR แทน TYPE_SYSTEM_OVERLAY อาจฟังดูเหมือนแฮ็ค แต่ให้คุณแสดงมุมมองด้านบนของทุกสิ่งและยังได้รับเหตุการณ์การสัมผัส


1
ฉันพยายามทุกอย่างเพื่อให้มันใช้งานได้ แต่ไม่สำเร็จ .... แต่การเพิ่ม TYPE_SYSTEM_ERROR ทำให้งานสำเร็จ .... ที่จริงแล้ว TYPE_SYSTEM_OVERLAY กำลังปิดกั้นการสัมผัส ..... + 1
Robi Kumar Tomar

6

TYPE_SYSTEM_OVERLAY ค่าคงนี้ถูกคัดค้านตั้งแต่ระดับ API 26 ใช้ TYPE_APPLICATION_OVERLAY แทน หรือ ** สำหรับผู้ใช้ด้านล่างและสูงกว่า android 8

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
    LAYOUT_FLAG = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
} else {
    LAYOUT_FLAG = WindowManager.LayoutParams.TYPE_PHONE;
}

5

ลองสิ่งนี้ ทำงานได้ดีใน 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

สิ่งนี้ใช้ไม่ได้สำหรับฉันหน้าจอของฉันไม่ทำงานกับพื้นหลัง วิธีการใด ๆ อื่น ๆ ที่คุณรู้ว่า ...
Vinuthan

@Vinuthan ตอบรับของคุณเป็นอย่างไรตอบ
Viswanath Lekshmanan

ไม่มันไม่ตอบสนอง ฉันเพิ่มกิจกรรมของฉันครึ่งหน้าจอเพื่อให้ฉันสามารถโต้ตอบกับพื้นหลังได้ แต่ฉันไม่สามารถโต้ตอบกับพื้นหลังได้
Vinuthan

FLAG_WATCH_OUTSIDE_TOUCH ทำให้คุณไม่สามารถโต้ตอบกับพื้นหลังได้
Richard Le Mesurier

1
TYPE_PHONE ใช้งานได้สำหรับฉัน - ตอนนี้มีการส่งมอบแบบสัมผัสแล้ว - อย่างน้อยฉันก็สามารถแตะปุ่ม;) ขอบคุณ!
vir พวกเรา

3

นี่คือวิธีแก้ปัญหาง่ายๆสิ่งที่คุณต้องมีก็คือการขยายเลย์เอาต์ 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;
    }

}

3

พบห้องสมุดที่ทำเช่นนั้น: https://github.com/recruit-lifestyle/FloatingView

มีโครงการตัวอย่างในโฟลเดอร์รูท ฉันวิ่งและใช้งานได้ตามต้องการ พื้นหลังสามารถคลิกได้ - แม้ว่าจะเป็นแอพอื่น

ป้อนคำอธิบายรูปภาพที่นี่


ขอบคุณสำหรับคำแนะนำฉันแค่พยายามค้นหาโครงการที่ยอดเยี่ยมนี้
upupming


1

ลองใช้รหัสของฉันอย่างน้อยก็ให้สตริงเป็นภาพซ้อนทับคุณสามารถแทนที่ด้วยปุ่มหรือรูปภาพได้ คุณจะไม่เชื่อว่านี่เป็นแอพสำหรับ Android ตัวแรกที่ฉันใช้ อย่างไรก็ตามถ้าคุณมีประสบการณ์กับแอพ Android มากกว่าฉันโปรดลอง

  • การเปลี่ยนพารามิเตอร์ 2 และ 3 ใน "New WindowManager.LayoutParams"
  • ลองใช้วิธีเหตุการณ์บางอย่างที่แตกต่างกัน

ฉันเวลาที่หาไม่ได้ในที่สุดก็จะไล่ตามนี้ด้วยเหตุนี้ฉันยังคงหายากเกี่ยวกับรายละเอียดของ ...
st0le

1

หากใครยังคงอ่านหัวข้อนี้และไม่สามารถทำงานนี้ได้ฉันเสียใจเป็นอย่างยิ่งที่จะบอกวิธีนี้ในการสกัดกั้นเหตุการณ์การเคลื่อนไหวถือเป็นข้อบกพร่องและแก้ไขใน Android> = 4.2

เหตุการณ์การเคลื่อนไหวที่คุณดักจับแม้ว่าจะมีการดำเนินการเป็น ACTION_OUTSIDE ให้ส่งคืน 0 ใน getX และ getY ซึ่งหมายความว่าคุณไม่สามารถเห็นตำแหน่งการเคลื่อนไหวทั้งหมดบนหน้าจอหรือคุณไม่สามารถทำอะไรได้ ฉันรู้ว่าหมอบอกว่ามันจะได้รับ x และ y แต่ความจริงก็คือมันจะไม่ ดูเหมือนว่าจะเป็นการบล็อคตัวบันทึกคีย์

หากใครมีวิธีแก้ปัญหาโปรดแสดงความคิดเห็นของคุณ

อ้างอิง: ทำไม ACTION_OUTSIDE ส่งคืน 0 ทุกครั้งใน KitKat 4.4.2

https://code.google.com/p/android/issues/detail?id=72746


1

โดยใช้บริการคุณสามารถบรรลุสิ่งนี้:

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;
        }
    }
}

1

@Sarwar Erfan คำตอบไม่ทำงานอีกต่อไปเนื่องจาก Android ไม่อนุญาตให้เพิ่มมุมมองด้วย WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY ไปที่หน้าต่างเพื่อให้สัมผัสได้อีกต่อไปไม่ได้กับแม้แต่ WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH

ฉันพบวิธีแก้ไขปัญหานี้แล้ว คุณสามารถตรวจสอบได้ในคำถามต่อไปนี้

เมื่อเพิ่มมุมมองไปที่หน้าต่างด้วย WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY จะไม่ได้รับเหตุการณ์การสัมผัส


วิธีแก้ปัญหาที่คุณอ้างถึงยังไม่สามารถใช้งานได้ใน Marshmallow
Ahmad Shahwaiz

@AhmadShahwaiz โปรดตรวจสอบลิงก์อีกครั้งฉันได้เพิ่มความคิดเห็นสำหรับปัญหาที่คุณพูดถึงที่นั่น
patilmandar2007

ใช่ตอนนี้มันทำงานอยู่ แต่ไม่ได้รับการตอบกลับจากการเรียกหลังจากเรียก startActivityForResult (เจตนา OVERLAY_REQ_CODE); ในโมฆะป้องกัน onActivityResult (int requestCode, int resultCode ข้อมูล Intent)
Ahmad Shahwaiz

@AhmadShahwaiz การเปลี่ยนแปลงที่กล่าวถึงในลิงค์จะต้องถูกเรียกโดยใช้กิจกรรมที่เราได้รับการติดต่อกลับในวิธี onActivityResult หากคุณกำลังตรวจสอบการอนุญาตข้างต้นในบริการคุณอาจต้องเพิ่มกิจกรรมแบบโปร่งใสที่จะเริ่มต้นจากบริการซึ่งสามารถขอสิทธิ์ผู้ใช้จากนั้นในกิจกรรมกิจกรรมตัวแทนเสมือนคุณจะได้รับผลลัพธ์จากผู้ใช้ หลังจากที่คุณจะต้องแจ้งบริการของคุณเพื่อเพิ่มมุมมองไปที่หน้าต่างหรือไม่ขึ้นอยู่กับการตอบสนองของผู้ใช้
patilmandar2007

1

นี่เป็นคำถามเก่า แต่เมื่อเร็ว ๆ นี้ Android มีการสนับสนุน Bubbles Bubbles กำลังจะเปิดตัวเร็ว ๆ นี้ แต่ปัจจุบันนักพัฒนาซอฟต์แวร์สามารถเริ่มใช้พวกเขาพวกมันถูกออกแบบมาให้ใช้SYSTEM_ALERT_WINDOWแทน แอพเช่น (Facebook Messenger และ MusiXMatch ใช้แนวคิดเดียวกัน)

ฟองฟอด

ฟองอากาศถูกสร้างขึ้นผ่าน API การแจ้งเตือนคุณส่งการแจ้งเตือนตามปกติ หากคุณต้องการให้ฟองคุณต้องแนบข้อมูลพิเศษบางอย่างไป สำหรับข้อมูลเพิ่มเติมเกี่ยวกับฟองคุณสามารถไปที่อย่างเป็นทางการของนักพัฒนาซอฟต์แวร์ Android คู่มือบนฟอง

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.