164 lines
5.9 KiB
C#
164 lines
5.9 KiB
C#
using System;
|
|
using System.Linq;
|
|
using Android.Content;
|
|
using Android.OS;
|
|
using Android.Runtime;
|
|
using Android.Service.Autofill;
|
|
using Android.Util;
|
|
using Android.Views.Autofill;
|
|
using Android.Widget;
|
|
using keepass2android.services.AutofillBase.model;
|
|
|
|
namespace keepass2android.services.AutofillBase
|
|
{
|
|
public interface IAutofillIntentBuilder
|
|
{
|
|
IntentSender GetAuthIntentSenderForResponse(Context context, string query, bool isManualRequest, bool autoReturnFromQuery);
|
|
Intent GetRestartAppIntent(Context context);
|
|
|
|
int AppIconResource { get; }
|
|
}
|
|
|
|
public abstract class AutofillServiceBase: AutofillService
|
|
{
|
|
public AutofillServiceBase()
|
|
{
|
|
|
|
}
|
|
|
|
public AutofillServiceBase(IntPtr javaReference, JniHandleOwnership transfer)
|
|
: base(javaReference, transfer)
|
|
{
|
|
}
|
|
|
|
|
|
public override void OnFillRequest(FillRequest request, CancellationSignal cancellationSignal, FillCallback callback)
|
|
{
|
|
bool isManual = (request.Flags & FillRequest.FlagManualRequest) != 0;
|
|
CommonUtil.logd( "onFillRequest " + (isManual ? "manual" : "auto"));
|
|
var structure = request.FillContexts[request.FillContexts.Count - 1].Structure;
|
|
|
|
//TODO support package signature verification as soon as this is supported in Keepass storage
|
|
|
|
var clientState = request.ClientState;
|
|
CommonUtil.logd( "onFillRequest(): data=" + CommonUtil.BundleToString(clientState));
|
|
|
|
|
|
cancellationSignal.CancelEvent += (sender, e) => {
|
|
Log.Warn(CommonUtil.Tag, "Cancel autofill not implemented yet.");
|
|
};
|
|
// Parse AutoFill data in Activity
|
|
string query = null;
|
|
var parser = new StructureParser(this, structure);
|
|
try
|
|
{
|
|
query = parser.ParseForFill(isManual);
|
|
|
|
}
|
|
catch (Java.Lang.SecurityException e)
|
|
{
|
|
Log.Warn(CommonUtil.Tag, "Security exception handling request");
|
|
callback.OnFailure(e.Message);
|
|
return;
|
|
}
|
|
|
|
AutofillFieldMetadataCollection autofillFields = parser.AutofillFields;
|
|
|
|
|
|
var autofillIds = autofillFields.GetAutofillIds();
|
|
if (autofillIds.Length != 0 && CanAutofill(query))
|
|
{
|
|
var responseBuilder = new FillResponse.Builder();
|
|
|
|
var entryDataset = AddEntryDataset(query, parser);
|
|
bool hasEntryDataset = entryDataset != null;
|
|
if (entryDataset != null)
|
|
responseBuilder.AddDataset(entryDataset);
|
|
|
|
AddQueryDataset(query, isManual, autofillIds, responseBuilder, !hasEntryDataset);
|
|
responseBuilder.SetSaveInfo(new SaveInfo.Builder(parser.AutofillFields.SaveType,
|
|
parser.AutofillFields.GetAutofillIds()).Build());
|
|
|
|
callback.OnSuccess(responseBuilder.Build());
|
|
}
|
|
else
|
|
{
|
|
callback.OnSuccess(null);
|
|
}
|
|
}
|
|
|
|
private Dataset AddEntryDataset(string query, StructureParser parser)
|
|
{
|
|
var filledAutofillFieldCollection = GetSuggestedEntry(query);
|
|
if (filledAutofillFieldCollection == null)
|
|
return null;
|
|
int partitionIndex = AutofillHintsHelper.GetPartitionIndex(parser.AutofillFields.FocusedAutofillCanonicalHints.FirstOrDefault());
|
|
FilledAutofillFieldCollection partitionData = AutofillHintsHelper.FilterForPartition(filledAutofillFieldCollection, partitionIndex);
|
|
|
|
return AutofillHelper.NewDataset(this, parser.AutofillFields, partitionData, IntentBuilder);
|
|
}
|
|
|
|
protected abstract FilledAutofillFieldCollection GetSuggestedEntry(string query);
|
|
|
|
private void AddQueryDataset(string query, bool isManual, AutofillId[] autofillIds, FillResponse.Builder responseBuilder, bool autoReturnFromQuery)
|
|
{
|
|
var sender = IntentBuilder.GetAuthIntentSenderForResponse(this, query, isManual, autoReturnFromQuery);
|
|
RemoteViews presentation = AutofillHelper.NewRemoteViews(PackageName,
|
|
GetString(Resource.String.autofill_sign_in_prompt), AppNames.LauncherIcon);
|
|
|
|
var datasetBuilder = new Dataset.Builder(presentation);
|
|
datasetBuilder.SetAuthentication(sender);
|
|
//need to add placeholders so we can directly fill after ChooseActivity
|
|
foreach (var autofillId in autofillIds)
|
|
{
|
|
datasetBuilder.SetValue(autofillId, AutofillValue.ForText("PLACEHOLDER"));
|
|
}
|
|
|
|
responseBuilder.AddDataset(datasetBuilder.Build());
|
|
}
|
|
|
|
private bool CanAutofill(string query)
|
|
{
|
|
return !(query == "androidapp://android" || query == "androidapp://"+this.PackageName);
|
|
}
|
|
|
|
public override void OnSaveRequest(SaveRequest request, SaveCallback callback)
|
|
{
|
|
|
|
var structure = request.FillContexts?.LastOrDefault()?.Structure;
|
|
if (structure == null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var parser = new StructureParser(this, structure);
|
|
string query = parser.ParseForSave();
|
|
try
|
|
{
|
|
HandleSaveRequest(parser, query);
|
|
callback.OnSuccess();
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
callback.OnFailure(e.Message);
|
|
}
|
|
|
|
}
|
|
|
|
protected abstract void HandleSaveRequest(StructureParser parser, string query);
|
|
|
|
|
|
public override void OnConnected()
|
|
{
|
|
CommonUtil.logd( "onConnected");
|
|
}
|
|
|
|
public override void OnDisconnected()
|
|
{
|
|
CommonUtil.logd( "onDisconnected");
|
|
}
|
|
|
|
public abstract IAutofillIntentBuilder IntentBuilder{get;}
|
|
}
|
|
}
|