如何解决 activity 无法从服务启动?

How to solve activity not starting from a service?

我有一个后台服务,它是一种无障碍服务(我认为这没什么区别)。我想从此服务启动一个新的 activity。我尝试使用这个建议的解决方案:

Intent dialogIntent = new Intent(this, ScreenshotActivity.class);
dialogIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
getApplicationContext().startActivity(dialogIntent);

然而,这似乎不起作用。我的 logcat 给出了这个输出,但是 activity 没有启动:

020-03-20 20:54:02.824 22977-22977/com.xx.xx I/Timeline: Timeline: Activity_launch_request time:19482980

我的服务和activity独立正常工作。但不是在这种情况下。也达到了我拨打此电话的起点。我用来测试的设备是 运行 Android 9.

编辑:主要服务代码:

public class MyAccessibility extends AccessibilityService {
public static MyAccessibility instance;
@Override
public void onCreate() {
    super.onCreate();
    Log.i("myLog", "create accessibility");
}

@Override
public void onAccessibilityEvent(AccessibilityEvent accessibilityEvent) {
    Log.i("myLog", "Event");
    if (instance == null) {
        Log.i("myLog", "Instance set to not null");
        instance = this;
    }

}

@Override
public void onInterrupt() {

}

@Override
protected void onServiceConnected() {
    super.onServiceConnected();

    AccessibilityServiceInfo info = new AccessibilityServiceInfo();
    info.eventTypes = AccessibilityEvent.TYPES_ALL_MASK;
    info.notificationTimeout = 100;
    info.feedbackType = AccessibilityServiceInfo.FEEDBACK_ALL_MASK;
    this.setServiceInfo(info);
    Toast t = Toast.makeText(getApplicationContext(), "Accessibility Service is connected now", Toast.LENGTH_SHORT);
    t.show();

    System.out.println("Accessibility was connected!");

    instance = this;

}

public void takeSS(){
    Intent dialogIntent = new Intent(this, ScreenshotActivity.class);
    dialogIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    this.startActivity(dialogIntent);
}

还有一些其他方法使用了 dispatchGesture。这就是为什么我有一个对此的静态引用,并且我可以从其他地方访问这些方法。截图activity使用媒体投影截屏并使用图像:

public class ScreenshotActivity extends Activity {
private static final int REQUEST_MEDIA_PROJECTION = 1;
private static final String TAG = "ScreenshotActivity";

private MediaProjectionManager mProjectionManager;

private MediaProjection mMediaProjection = null;

private VirtualDisplay mVirtualDisplay;

private ImageReader mImageReader;

private static final int MAX_IMAGE_BUFFER = 10;
private  int counter;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    setContentView(R.layout.activity_screenshot_dummy);

    counter = 0;

    OrientationChangedListener mOrientationChangedListener = new OrientationChangedListener(this);
    mOrientationChangedListener.enable();

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
        mProjectionManager = (MediaProjectionManager)getSystemService(MEDIA_PROJECTION_SERVICE);
        startActivityForResult(mProjectionManager.createScreenCaptureIntent(), REQUEST_MEDIA_PROJECTION);
    }
}

@TargetApi(Build.VERSION_CODES.LOLLIPOP)
public void onActivityResult(int requestCode, int resultCode, Intent resultData) {
    super.onActivityResult(requestCode, resultCode, resultData);
    if (requestCode == REQUEST_MEDIA_PROJECTION) {
        String message;

        if (resultCode != Activity.RESULT_OK) {
            message = "Media Projection Declined";
            mMediaProjection = null;
        } else {

            message = "Media Projection Accepted";
            mMediaProjection = mProjectionManager.getMediaProjection(resultCode, resultData);
            attachImageCaptureOverlay();

        }

        Toast toast = Toast.makeText(this, message, Toast.LENGTH_SHORT);
        toast.show();

    }
}

private class OrientationChangedListener extends OrientationEventListener {

    int mLastOrientation = -1;

    OrientationChangedListener(Context context) {
        super(context);
    }

    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    @Override
    public void onOrientationChanged(int orientation) {

        final int screenOrientation = getWindowManager().getDefaultDisplay().getRotation();

        if (mVirtualDisplay == null) return;

        if (mLastOrientation == screenOrientation) return;

        mLastOrientation = screenOrientation;

        detachImageCaptureOverlay();

        attachImageCaptureOverlay();
    }
}

private final ImageReader.OnImageAvailableListener mOnImageAvailableListener = new ImageReader.OnImageAvailableListener() {

    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    @Override
    public void onImageAvailable(ImageReader reader) {

        Image image = reader.acquireLatestImage();

        if (image == null || image.getPlanes().length <= 0) return;

        final Image.Plane plane = image.getPlanes()[0];

        final int rowPadding = plane.getRowStride() - plane.getPixelStride() * image.getWidth();
        final int bitmapWidth = image.getWidth() + rowPadding / plane.getPixelStride();

        final Bitmap tempBitmap = Bitmap.createBitmap(bitmapWidth, image.getHeight(), Bitmap.Config.ARGB_8888);
        tempBitmap.copyPixelsFromBuffer(plane.getBuffer());

        Rect cropRect = image.getCropRect();
        final Bitmap bitmap = Bitmap.createBitmap(tempBitmap, cropRect.left, cropRect.top, cropRect.width(), cropRect.height());

        //Do something with the bitmap


        image.close();
    }
};

@RequiresApi(Build.VERSION_CODES.LOLLIPOP)
private void attachImageCaptureOverlay() {

    if (mMediaProjection == null) return;

    final DisplayMetrics metrics = new DisplayMetrics();
    getWindowManager().getDefaultDisplay().getRealMetrics(metrics);

    mImageReader = ImageReader.newInstance(metrics.widthPixels, metrics.heightPixels, PixelFormat.RGBA_8888, MAX_IMAGE_BUFFER);

    mVirtualDisplay = mMediaProjection.createVirtualDisplay("ScreenCaptureTest",
            metrics.widthPixels, metrics.heightPixels, metrics.densityDpi,
            DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,
            mImageReader.getSurface(), null, null);

    mImageReader.setOnImageAvailableListener(mOnImageAvailableListener, null);
}

@RequiresApi(api = Build.VERSION_CODES.KITKAT)
private void detachImageCaptureOverlay() {
    mVirtualDisplay.release();
    mImageReader.close();
}


}

您无法从后台启动 activity since Android Oreo

一种解决方法是在您的服务启动时调用 startForeground(true),然后通过适当的操作将粘性 Notification 添加到您的服务 (starting your desired activity via PendingIntent)。