- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我是密码管理器应用的独立开发者。我应该怎么做,或者我应该实现什么(接口(interface)/API/服务),使我的应用程序成为自动填充服务提供商(在 Android Oreo API >= 26 的设备中)?
我看了各种相关的文档,但是就是看不懂怎么弄。我错过了什么吗?
目前我看到只有知名的密码管理器支持这个功能:
欢迎任何提示。
最佳答案
像往常一样,谷歌自己的examples repository provides a good starting point用于学习 Autofill Framework 的 API,并且涵盖的 Material 比我能放入答案的 Material 多得多。下面是关键概念的概述。来自 documentation 中的描述,我们想要创建一个自动填充服务,它将处理来自其他应用程序(客户端)的请求以存储和检索自动填充字段数据。
首先,我们需要创建一个服务提供者类来履行这个契约。我们可以扩展基地 AutofillService
class :
import android.service.autofill.AutofillService;
...
public class MyAutofillService extends AutofillService {
...
@Override
public void onFillRequest(FillRequest request, CancellationSignal cancellationSignal,
FillCallback callback) { ... }
@Override
public void onSaveRequest(SaveRequest request, SaveCallback callback) { ... }
}
服务的 onFillRequest()
和 onSaveRequest()
方法对我们的理解最重要。 Android系统调用onFillRequest()
确定我们的服务是否可以为特定 Activity 自动填充字段,并为方法提供 FillRequest
其中包含我们的服务将检查可填写字段的上下文和 View 信息。服务完成后,它会调用提供的 callback
使用适当的自动填充数据。
这是为 FillRequest
提供自动填充建议所需的基本步骤的显着简化概述。 :
@Override
public void onFillRequest(FillRequest request, CancellationSignal signal, FillCallback callback) {
List<FillContext> contexts = request.getFillContexts();
AssistStructure structure = contexts.get(contexts.size() - 1);
WindowNode windowNode = structure.getWindowNodeAt(0);
ViewNode viewNode = windowNode.getRootViewNode(); // pretend this is an EditText
String suggestionText = "This will appear in the autofill list for 'viewNode'.";
RemoteViews suggestion = new RemoteViews(getPackageName(), R.layout.autofill_suggestion);
suggestion.setTextViewText(R.id.autofill_suggestion, suggestionText);
Dataset suggestionDataset = new Dataset.Builder(suggestion)
.setValue(viewNode.getAutoFillId(), AutofillValue.forText(suggestionText))
.build();
FillResponse response = new FillResponse.Builder()
.addDataset(suggestionDataset)
.build();
callback.onSuccess(response);
}
正如我们所见,自动填充 API 需要大量代码才能为 View
提供单个静态自动填充建议。我们已经提前知道了——这个例子假设viewNode
是我们要为其提供自动填充建议的文本输入字段。实际上,这个例子太简单了,但我想清楚地展示最小实现。对于每个 WindowNode
,我们需要遍历根的 View 树 ViewNode
及其每个 child 找到我们的服务希望为其提供自动填充数据的每个输入字段,然后创建一个 RemoteViews
和 Dataset
其中包含我们将添加到 FillResponse
的每个字段的自动填充建议使用 FillResponse.Builder.addDataset()
.此示例未显示 R.layout.autofill_suggestion
的纯 XML 布局TextView
用于为 RemoteViews
创建建议显示项.
同样,Android调用onSaveRequest()
当用户想要将数据保存在 Activity 的字段中以供将来完成请求并注入(inject) SaveRequest
时我们的服务使用它来查找要记住的自动填充数据。
这些方法中的每一个的具体实现将取决于我们的应用程序提供的自动填充数据的类型。自动填充服务必须认真检查每个字段的特征并仔细选择一组适当的自动填充建议以避免将用户数据泄露给恶意客户端 Activity (参见评论)。特别是对于密码管理器,我们需要特别注意正确验证服务的用户并在请求和保存自动填充数据时提供一组安全的建议。
我们现在可以 register the service在<application>
项目的 AndroidManifest.xml block :
<service
android:name=".MyAutofillService"
android:label="Multi-Dataset Autofill Service"
android:permission="android.permission.BIND_AUTOFILL_SERVICE">
<meta-data
android:name="android.autofill"
android:resource="@xml/multidataset_service" />
<intent-filter>
<action android:name="android.service.autofill.AutofillService" />
</intent-filter>
</service>
如图所示,这会将我们的自动填充服务绑定(bind)为出现在问题中显示的自动填充服务列表中的可用选项。 android:name
属性必须匹配我们的名称 AutofillService
类,我们的应用程序必须声明 BIND_AUTOFILL_SERVICE
允许。更改 android:label
的值为该服务取一个合适的名称(例如,“Autofill with Password Manager”)。或者,在字符串资源中设置它。另请注意,我们应该提供一个“设置” Activity ,用于配置我们在 <meta‑data>
中指定的服务。对于 android.autofill
:
<autofill-service android:settingsActivity="foo.bar.SettingsActivity" />
然后,用户可以从他们的设备设置中启用我们的自动填充服务。我们可以广播 ACTION_REQUEST_SET_AUTOFILL_SERVICE
在设置或首次启动期间帮助用户找到此屏幕的 Intent 。
关于安卓奥利奥 : what should I do to publish my app as an Autofill service provider?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47367576/
正在复制的问题是,当sew呈现登录页面时,然后我们继续执行身份验证,现在将我们重定向到网站的仪表板,此时我们继续关闭会话,并且在之前的交互中生成的这些cookie没有被删除。这个问题是重复性的,并且一
我是 Flutter 的新手,目前正在研究 DI。 我正在使用 flutter_bloc 和 provider 包。 flutter_bloc 附带一个 RepositoryProvider,我现在问
我是 Flutter 的新手,目前正在研究 DI。 我正在使用 flutter_bloc 和 provider 包。 flutter_bloc 附带一个 RepositoryProvider,我现在问
我正在使用 Angular2 开发一个应用程序。 我正在尝试在我的应用程序中使用 Reactive Forms,但我遇到了一些错误: 第一个错误是关于 NgControl 的,如下所示: No pro
最近很多用户在使用电脑的时候发现了wmi provider host进程占用内存比较大,不知道这个进程到底是干什么的,能不能禁止,怎么禁止。下面来一起看看想想的介绍吧。 wmi provide
我的问题是: 当我在设计时不知道这些表达式的数量和类型时,如何将列表中的表达式拼接成一个引用? 在底部,我包含了类型提供程序的完整代码。 (我已经剥离了这个概念来证明这个问题。)我的问题出现在这些行:
我目前正在学习使用 Flutter 进行应用程序开发,并已开始学习 Provider 包。我遇到了一些困难并收到错误: “在此...小部件之上找不到正确的提供者” 我最终移动了 Provider 小部
我是 android 的新手,我正在学习如何使用 JavaMail API 发送电子邮件的教程,我已经正确添加了必要的 Jar,但我总是遇到无法解析 GmailSender 类上的符号提供程序,我尝试
我正在我的 Angular 应用程序中进行单元测试,我正在使用 TestBed 方法, 我正在测试组件,所以每个规范文件看起来像这样 import... describe('AppComponent'
enter image description here 代码:这是我的 index.js 文件 index.js import { Provider } from "react-redux"
Microsoft ASP.NET Universal Providers 1.1昨天与System.Web.Providers 1.2一起发布.在后面的 nuget 页面上声明:Legacy pac
在我的 Next js 项目中,我使用了 Next auth,其中 import {Provider} from 'next-auth/client' , 并包裹 在 _app.js 中。 但是,与此
当我在 View 模型中使用如下界面时 class MainViewModel @ViewModelInject constructor( private val trafficImagesR
更新 - 我实际上发现它是 Flutter Issue . 我有两个 Provider,一个是 EntriesProvider,另一个是 EntryProvider。我在创建条目时使用我的 Entry
function configure($provide, $injector) { $provide.provider("testservice", function () {
这真让我抓狂。我似乎无法弄清楚这有什么问题。 代码: public interface IMinutesCounter { void startTimer(); void stopTi
我在我的项目中玩 Dagger 2,然后我陷入了这个错误编译。-> Error:(18, 21) error: ....MyManager cannot be provided without an
我有一个 Resteasy 应用程序,它使用 Spring 并包含 ContainerRequestFilter 和 ContainerResponseFilter 实现,并用 @Provider 注
我正在尝试使用 Dagger2 设置一个新项目,我以前使用过 Dagger2,但现在我正在尝试自己从头开始设置它。我正在从我参与的 Kotlin 项目中获取示例,但无法像现在在 Kotlin 中一样为
我刚开始学习 dagger2,遇到了一个奇怪的问题,在我看来像是一个错误。这是模块: @Module public class SimpleModule { @Provides Coo
我是一名优秀的程序员,十分优秀!