2020-01-09 20:17:16 +03:00
|
|
|
|
using System;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Android.App;
|
|
|
|
|
using Android.Content;
|
2020-01-09 20:17:16 +03:00
|
|
|
|
using Android.Graphics;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
using Android.OS;
|
2020-01-09 20:17:16 +03:00
|
|
|
|
using Android.Provider;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
using Android.Runtime;
|
2020-01-09 20:17:16 +03:00
|
|
|
|
using Android.Views;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
using Android.Views.Accessibility;
|
2020-01-09 20:17:16 +03:00
|
|
|
|
using Android.Widget;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
using Bit.App.Resources;
|
|
|
|
|
using Bit.Core;
|
|
|
|
|
using Bit.Core.Abstractions;
|
|
|
|
|
using Bit.Core.Utilities;
|
2020-01-28 01:36:20 +03:00
|
|
|
|
using Java.Util;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
|
|
|
|
|
namespace Bit.Droid.Accessibility
|
|
|
|
|
{
|
|
|
|
|
[Service(Permission = Android.Manifest.Permission.BindAccessibilityService, Label = "Bitwarden")]
|
|
|
|
|
[IntentFilter(new string[] { "android.accessibilityservice.AccessibilityService" })]
|
|
|
|
|
[MetaData("android.accessibilityservice", Resource = "@xml/accessibilityservice")]
|
|
|
|
|
[Register("com.x8bit.bitwarden.Accessibility.AccessibilityService")]
|
|
|
|
|
public class AccessibilityService : Android.AccessibilityServices.AccessibilityService
|
|
|
|
|
{
|
|
|
|
|
private const string BitwardenPackage = "com.x8bit.bitwarden";
|
|
|
|
|
private const string BitwardenWebsite = "vault.bitwarden.com";
|
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
private AccessibilityNodeInfo _anchorNode = null;
|
|
|
|
|
private int _lastAnchorX, _lastAnchorY = 0;
|
|
|
|
|
private static bool _overlayAnchorObserverRunning = false;
|
|
|
|
|
private IWindowManager _windowManager = null;
|
|
|
|
|
private LinearLayout _overlayView = null;
|
|
|
|
|
private long _lastAutoFillTime = 0;
|
|
|
|
|
private Java.Lang.Runnable _overlayAnchorObserverRunnable = null;
|
|
|
|
|
private Handler _handler = new Handler(Looper.MainLooper);
|
2020-01-09 20:17:16 +03:00
|
|
|
|
|
2019-04-30 21:33:00 +03:00
|
|
|
|
private HashSet<string> _launcherPackageNames = null;
|
|
|
|
|
private DateTime? _lastLauncherSetBuilt = null;
|
|
|
|
|
private TimeSpan _rebuildLauncherSpan = TimeSpan.FromHours(1);
|
|
|
|
|
|
|
|
|
|
public override void OnAccessibilityEvent(AccessibilityEvent e)
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var powerManager = GetSystemService(PowerService) as PowerManager;
|
|
|
|
|
if(Build.VERSION.SdkInt > BuildVersionCodes.KitkatWatch && !powerManager.IsInteractive)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
else if(Build.VERSION.SdkInt < BuildVersionCodes.Lollipop && !powerManager.IsScreenOn)
|
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(SkipPackage(e?.PackageName))
|
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(e?.PackageName != "com.android.systemui")
|
|
|
|
|
{
|
|
|
|
|
CancelOverlayPrompt();
|
|
|
|
|
}
|
2019-04-30 21:33:00 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-09 20:17:16 +03:00
|
|
|
|
// AccessibilityHelpers.PrintTestData(root, e);
|
2019-04-30 21:33:00 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
AccessibilityNodeInfo root = null;
|
|
|
|
|
|
2019-04-30 21:33:00 +03:00
|
|
|
|
switch(e.EventType)
|
|
|
|
|
{
|
|
|
|
|
case EventTypes.ViewFocused:
|
2020-01-09 20:17:16 +03:00
|
|
|
|
case EventTypes.ViewClicked:
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(e.Source == null || e.PackageName == BitwardenPackage)
|
|
|
|
|
{
|
|
|
|
|
CancelOverlayPrompt();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
root = RootInActiveWindow;
|
|
|
|
|
if(root == null || root.PackageName != e.PackageName)
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-01-09 21:17:17 +03:00
|
|
|
|
var isKnownBroswer = AccessibilityHelpers.SupportedBrowsers.ContainsKey(root.PackageName);
|
|
|
|
|
if(e.EventType == EventTypes.ViewClicked && isKnownBroswer)
|
2020-01-09 20:17:16 +03:00
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(!(e.Source?.Password ?? false) && !AccessibilityHelpers.IsUsernameEditText(root, e))
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-09 20:17:16 +03:00
|
|
|
|
CancelOverlayPrompt();
|
2019-04-30 21:33:00 +03:00
|
|
|
|
break;
|
|
|
|
|
}
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(ScanAndAutofill(root, e))
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
CancelOverlayPrompt();
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
2020-01-09 20:17:16 +03:00
|
|
|
|
else
|
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
OverlayPromptToAutofill(root, e);
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case EventTypes.WindowContentChanged:
|
|
|
|
|
case EventTypes.WindowStateChanged:
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(AccessibilityHelpers.LastCredentials == null)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(e.PackageName == BitwardenPackage)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
CancelOverlayPrompt();
|
2019-04-30 21:33:00 +03:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
root = RootInActiveWindow;
|
|
|
|
|
if(root == null || root.PackageName != e.PackageName)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-01-09 20:17:16 +03:00
|
|
|
|
if(ScanAndAutofill(root, e))
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-09 20:17:16 +03:00
|
|
|
|
CancelOverlayPrompt();
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Suppress exceptions so that service doesn't crash.
|
2020-01-09 21:17:17 +03:00
|
|
|
|
catch(Exception ex)
|
2020-01-09 20:17:16 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
System.Diagnostics.Debug.WriteLine(">>> {0}: {1}", ex.GetType(), ex.StackTrace);
|
2020-01-09 20:17:16 +03:00
|
|
|
|
}
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void OnInterrupt()
|
|
|
|
|
{
|
|
|
|
|
// Do nothing.
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-09 20:17:16 +03:00
|
|
|
|
public bool ScanAndAutofill(AccessibilityNodeInfo root, AccessibilityEvent e)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-09 20:17:16 +03:00
|
|
|
|
var filled = false;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
var passwordNodes = AccessibilityHelpers.GetWindowNodes(root, e, n => n.Password, false);
|
|
|
|
|
if(passwordNodes.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
var uri = AccessibilityHelpers.GetUri(root);
|
|
|
|
|
if(uri != null && !uri.Contains(BitwardenWebsite))
|
|
|
|
|
{
|
|
|
|
|
if(AccessibilityHelpers.NeedToAutofill(AccessibilityHelpers.LastCredentials, uri))
|
|
|
|
|
{
|
|
|
|
|
AccessibilityHelpers.GetNodesAndFill(root, e, passwordNodes);
|
2020-01-09 20:17:16 +03:00
|
|
|
|
filled = true;
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_lastAutoFillTime = Java.Lang.JavaSystem.CurrentTimeMillis();
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
AccessibilityHelpers.LastCredentials = null;
|
|
|
|
|
}
|
|
|
|
|
else if(AccessibilityHelpers.LastCredentials != null)
|
|
|
|
|
{
|
|
|
|
|
Task.Run(async () =>
|
|
|
|
|
{
|
|
|
|
|
await Task.Delay(1000);
|
|
|
|
|
AccessibilityHelpers.LastCredentials = null;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
passwordNodes.Dispose();
|
2020-01-09 20:17:16 +03:00
|
|
|
|
return filled;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-09 20:17:16 +03:00
|
|
|
|
private void CancelOverlayPrompt()
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_overlayAnchorObserverRunning = false;
|
|
|
|
|
|
|
|
|
|
if(_windowManager != null && _overlayView != null)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_windowManager.RemoveViewImmediate(_overlayView);
|
|
|
|
|
System.Diagnostics.Debug.WriteLine(">>> Accessibility Overlay View Removed");
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
2020-01-09 20:17:16 +03:00
|
|
|
|
|
|
|
|
|
_overlayView = null;
|
2020-01-14 01:14:57 +03:00
|
|
|
|
_lastAnchorX = 0;
|
|
|
|
|
_lastAnchorY = 0;
|
2020-01-28 01:36:20 +03:00
|
|
|
|
|
|
|
|
|
if(_anchorNode != null)
|
|
|
|
|
{
|
|
|
|
|
_anchorNode.Recycle();
|
|
|
|
|
_anchorNode = null;
|
|
|
|
|
}
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-09 20:17:16 +03:00
|
|
|
|
private void OverlayPromptToAutofill(AccessibilityNodeInfo root, AccessibilityEvent e)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-10 18:20:19 +03:00
|
|
|
|
if(!AccessibilityHelpers.OverlayPermitted())
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-09 20:17:16 +03:00
|
|
|
|
System.Diagnostics.Debug.WriteLine(">>> Overlay Permission not granted");
|
|
|
|
|
Toast.MakeText(this, AppResources.AccessibilityOverlayPermissionAlert, ToastLength.Long).Show();
|
2020-01-28 01:36:20 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(_overlayView != null || _anchorNode != null || _overlayAnchorObserverRunning)
|
|
|
|
|
{
|
|
|
|
|
CancelOverlayPrompt();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(Java.Lang.JavaSystem.CurrentTimeMillis() - _lastAutoFillTime < 1000)
|
|
|
|
|
{
|
2019-04-30 21:33:00 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-09 20:17:16 +03:00
|
|
|
|
var uri = AccessibilityHelpers.GetUri(root);
|
2020-01-09 21:17:17 +03:00
|
|
|
|
if(string.IsNullOrWhiteSpace(uri))
|
2020-01-09 20:17:16 +03:00
|
|
|
|
{
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-14 01:14:57 +03:00
|
|
|
|
var layoutParams = AccessibilityHelpers.GetOverlayLayoutParams();
|
|
|
|
|
var anchorPosition = AccessibilityHelpers.GetOverlayAnchorPosition(root, e.Source);
|
2020-01-09 20:17:16 +03:00
|
|
|
|
layoutParams.X = anchorPosition.X;
|
|
|
|
|
layoutParams.Y = anchorPosition.Y;
|
|
|
|
|
|
2020-01-09 21:17:17 +03:00
|
|
|
|
if(_windowManager == null)
|
2020-01-09 20:17:16 +03:00
|
|
|
|
{
|
|
|
|
|
_windowManager = GetSystemService(WindowService).JavaCast<IWindowManager>();
|
|
|
|
|
}
|
2019-04-30 21:33:00 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
var intent = new Intent(this, typeof(AccessibilityActivity));
|
|
|
|
|
intent.PutExtra("uri", uri);
|
|
|
|
|
intent.SetFlags(ActivityFlags.NewTask | ActivityFlags.SingleTop | ActivityFlags.ClearTop);
|
|
|
|
|
|
|
|
|
|
_overlayView = AccessibilityHelpers.GetOverlayView(this);
|
|
|
|
|
_overlayView.Click += (sender, eventArgs) =>
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
CancelOverlayPrompt();
|
|
|
|
|
StartActivity(intent);
|
|
|
|
|
};
|
2020-01-09 20:17:16 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_anchorNode = e.Source;
|
|
|
|
|
_lastAnchorX = anchorPosition.X;
|
|
|
|
|
_lastAnchorY = anchorPosition.Y;
|
2020-01-14 01:14:57 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_windowManager.AddView(_overlayView, layoutParams);
|
2020-01-09 20:17:16 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
System.Diagnostics.Debug.WriteLine(">>> Accessibility Overlay View Added at X:{0} Y:{1}",
|
|
|
|
|
layoutParams.X, layoutParams.Y);
|
2020-01-09 20:17:16 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
StartOverlayAnchorObserver();
|
|
|
|
|
}
|
2020-01-14 01:14:57 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
private void StartOverlayAnchorObserver()
|
|
|
|
|
{
|
|
|
|
|
if(_overlayAnchorObserverRunning)
|
|
|
|
|
{
|
|
|
|
|
return;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_overlayAnchorObserverRunning = true;
|
2020-01-14 01:14:57 +03:00
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_overlayAnchorObserverRunnable = new Java.Lang.Runnable(() =>
|
|
|
|
|
{
|
|
|
|
|
if(_overlayAnchorObserverRunning)
|
|
|
|
|
{
|
|
|
|
|
AdjustOverlayForScroll();
|
|
|
|
|
_handler.PostDelayed(_overlayAnchorObserverRunnable, 250);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
_handler.PostDelayed(_overlayAnchorObserverRunnable, 250);
|
2020-01-14 01:14:57 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
private void AdjustOverlayForScroll()
|
2020-01-14 01:14:57 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(_overlayView == null || _anchorNode == null)
|
2019-04-30 21:33:00 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
CancelOverlayPrompt();
|
2020-01-14 01:14:57 +03:00
|
|
|
|
return;
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
var root = RootInActiveWindow;
|
|
|
|
|
IEnumerable<AccessibilityWindowInfo> windows = null;
|
|
|
|
|
if(Build.VERSION.SdkInt > BuildVersionCodes.Kitkat)
|
|
|
|
|
{
|
|
|
|
|
windows = Windows;
|
|
|
|
|
}
|
|
|
|
|
var anchorPosition = AccessibilityHelpers.GetOverlayAnchorPosition(_anchorNode, root, windows);
|
|
|
|
|
|
2020-01-14 01:14:57 +03:00
|
|
|
|
if(anchorPosition == null)
|
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
CancelOverlayPrompt();
|
2020-01-14 01:14:57 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
2020-01-28 01:36:20 +03:00
|
|
|
|
else if(anchorPosition.X == -1 && anchorPosition.Y == -1)
|
2020-01-14 01:14:57 +03:00
|
|
|
|
{
|
2020-01-28 01:36:20 +03:00
|
|
|
|
if(_overlayView.Visibility != ViewStates.Gone)
|
|
|
|
|
{
|
|
|
|
|
_overlayView.Visibility = ViewStates.Gone;
|
|
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
else if(anchorPosition.X == _lastAnchorX && anchorPosition.Y == _lastAnchorY)
|
|
|
|
|
{
|
|
|
|
|
if(_overlayView.Visibility != ViewStates.Visible)
|
|
|
|
|
{
|
|
|
|
|
_overlayView.Visibility = ViewStates.Visible;
|
|
|
|
|
}
|
2020-01-14 01:14:57 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var layoutParams = AccessibilityHelpers.GetOverlayLayoutParams();
|
|
|
|
|
layoutParams.X = anchorPosition.X;
|
|
|
|
|
layoutParams.Y = anchorPosition.Y;
|
|
|
|
|
|
|
|
|
|
_lastAnchorX = anchorPosition.X;
|
|
|
|
|
_lastAnchorY = anchorPosition.Y;
|
|
|
|
|
|
2020-01-28 01:36:20 +03:00
|
|
|
|
_windowManager.UpdateViewLayout(_overlayView, layoutParams);
|
|
|
|
|
|
|
|
|
|
if(_overlayView.Visibility != ViewStates.Visible)
|
|
|
|
|
{
|
|
|
|
|
_overlayView.Visibility = ViewStates.Visible;
|
|
|
|
|
}
|
|
|
|
|
|
2020-01-14 01:14:57 +03:00
|
|
|
|
System.Diagnostics.Debug.WriteLine(">>> Accessibility Overlay View Updated to X:{0} Y:{1}",
|
|
|
|
|
layoutParams.X, layoutParams.Y);
|
2019-04-30 21:33:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool SkipPackage(string eventPackageName)
|
|
|
|
|
{
|
|
|
|
|
if(string.IsNullOrWhiteSpace(eventPackageName) ||
|
|
|
|
|
AccessibilityHelpers.FilteredPackageNames.Contains(eventPackageName) ||
|
|
|
|
|
eventPackageName.Contains("launcher"))
|
|
|
|
|
{
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if(_launcherPackageNames == null || _lastLauncherSetBuilt == null ||
|
|
|
|
|
(DateTime.Now - _lastLauncherSetBuilt.Value) > _rebuildLauncherSpan)
|
|
|
|
|
{
|
|
|
|
|
// refresh launcher list every now and then
|
|
|
|
|
_lastLauncherSetBuilt = DateTime.Now;
|
|
|
|
|
var intent = new Intent(Intent.ActionMain);
|
|
|
|
|
intent.AddCategory(Intent.CategoryHome);
|
|
|
|
|
var resolveInfo = PackageManager.QueryIntentActivities(intent, 0);
|
|
|
|
|
_launcherPackageNames = resolveInfo.Select(ri => ri.ActivityInfo.PackageName).ToHashSet();
|
|
|
|
|
}
|
|
|
|
|
return _launcherPackageNames.Contains(eventPackageName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|