- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在使用 AutoCompleteTextView 让我的 Android 应用程序的用户使用自定义网络服务搜索内容:它按预期工作,但目前我找不到实现“无尽滚动”的方法"在下拉 ListView 上。
现在,我的 AutoCompleteTextView 适配器是一个实现了 Filterable 接口(interface)的 ArrayAdapter;每当用户更改 AutoCompleteTextView 的文本时,我的过滤器的 performFiltering() 方法就会被触发,我可以向我的自定义 Web 服务发出 HTTP 请求以显示适当的内容。但我想在用户滚动下拉菜单时加载更多内容,这是一个分页系统,这样我就可以避免一次加载数百个结果……我不知道该怎么做!< br/>
谢谢大家:)
我的 fragment 代码
AutoCompleteTextView search = (AutoCompleteTextView) view.findViewById(R.id.search);
SearchAdapter searchAdapter = new SearchAdapter(getActivity(), 0);
search.setAdapter(searchAdapter);
我的适配器代码(已编辑)
class SearchAdapter extends ArrayAdapter<Parcelable> implements Filterable {
Integer numberPerPage = 10;
Boolean moreDataIsAvailable = false;
FragmentActivity activity;
public ArrayList<Parcelable> items = new ArrayList<Parcelable>();
public SearchAdapter(FragmentActivity a, int textViewResourceId) {
super(a, textViewResourceId);
activity = a;
}
@Override
public int getCount() {
return items.size();
}
@Override
public Parcelable getItem(int index) {
if(items.size() > index) {
return items.get(index);
} else {
return null;
}
}
@Override
public Filter getFilter() {
Filter filter = new Filter() {
@Override
protected FilterResults performFiltering(CharSequence constraint) {
FilterResults filterResults = new FilterResults();
if (constraint != null && constraint.toString().length() >= 3) {
autocomplete(constraint.toString(), items);
filterResults.count = items.size();
filterResults.values = items;
} else {
items.clear();
filterResults.count = items.size();
filterResults.values = items;
}
return filterResults;
}
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
if (results != null && results.count > 0) {
notifyDataSetChanged();
} else {
notifyDataSetInvalidated();
}
}
};
return filter;
}
static class ViewHolder {
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
View rowView = convertView;
if (rowView == null) {
rowView = LayoutInflater.from(activity).inflate(R.layout.search_row, parent, false);
holder = new ViewHolder();
rowView.setTag(holder);
} else {
holder = (ViewHolder) rowView.getTag();
}
// Setting my row data
return rowView;
}
private void autocomplete(String input, ArrayList<Parcelable> items) {
ArrayList<Parcelable> data = new ArrayList<Parcelable>();
try {
RequestHandler request = new RequestHandler();
JSONObject requestParameters = new JSONObject();
requestParameters.put("offset", 0);
requestParameters.put("keyword", input);
requestParameters.put("limit", numberPerPage);
ResponseDescription response = request.request(activity, requestParameters);
if(!response.error) {
JSONArray searchedItems = response.getJSONArray("items");
if(searchedItems.length() == numberPerPage) {
moreDataIsAvailable = true;
} else {
moreDataIsAvailable = false;
}
for(int i = 0 ; i < searchedItems.length(); i++) {
JSONObject searchedItem = searchedItems.getJSONObject(i);
MyObject object = new MyObject();
object.initWithJSONObject(searchedItem);
data.add(object);
}
}
} catch (Exception e) {
e.printStackTrace();
}
items.clear();
items.addAll(data);
}
}
最佳答案
不幸的是,没有用于自动完成的 OnScrollListener
接口(interface),但我认为您可以解决这个问题。
这是我认为的食谱:
autocomplete
方法添加另一个 argument 参数。我注意到您将偏移量硬编码为“0”。您将希望能够使用相应的值调用 autocomplete
。此外,您在 autocomplete
的末尾有 items.clear()
并且您只想在偏移量为零时执行此操作。
AsyncTask
。或者除了 Filter
中的 performFiltering
方法之外,您还可以在后台运行 autocomplete
方法。此异步任务需要访问您的适配器,以便它可以将其结果添加到 items
列表并像过滤器一样调用 notifyDataSetChanged()
。
您的适配器需要保留对此任务的引用,以便您可以在用户再次开始输入时取消它。
getView()
添加一些逻辑以执行异步任务。我们没有 OnScrollListener
,所以我们将使用 getView()
作为它的一种代理。
为启动下一个请求设置一个常量阈值。它需要小于您的 numberPerPage
,所以我们以“5”为例。
现在,当 ListView
调用 getView()
并从末尾开始在阈值范围内的位置时,执行异步任务。例如,在第一次筛选操作之后,您的列表中有 10 个项目。当 ListView
请求第 5 项的 View 时,使用等于您的列表大小的偏移量启动异步任务 - 在本例中为 10(以获取第 11-20 项)。
我的假设是 ListView
只会在用户向下滚动到该项目时请求项目 View 。
performFiltering()
中,取消任何正在运行的异步任务。我已经完成了这种类型的“无限”滚动,使用 ListView
从 OnScrollListener
调用 AsyncTask
以逐页返回搜索结果,我认为它对我有用的原因是请求响应包含总大小,因此我能够将其用于 getCount()
。
我还使用了自动完成功能来获取远程结果,但我不必对这些结果进行分页。事实上,我想我只是在输入发生变化并且我的过滤器方法是空操作时运行了异步任务。
结合自动完成和分页的情况不同。如果用户在异步任务可以更新它之前到达列表末尾,则滚动下拉列表可能会出现一些卡顿行为。因此,您可能需要调整页面大小、阈值甚至 getCount()
的返回值以获得可接受的滚动体验。但我认为这是可行的。
关于安卓 : AutoCompleteTextView listview scroll listener (for endless scrolling),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29276545/
我想将 EditText 始终设置为 LTR,我该怎么做?android:textDirection 似乎确实是答案,我无法让项目使用该指令进行编译 最佳答案 为 editText 使用引力 andr
我希望我的应用在每次打开时都显示登录屏幕。使用 android:clearTaskOnLaunch="true" 一切正常。但是有一个错误,只有在手机关机并且应用程序首先使用小部件启动时才会发生。 I
实际上我的要求是我想将动态文件加载到像图像一样的网页中。视频,音频等,它来自 Assets 或应用程序自己的文件目录,这不是问题。 我尝试使用以下两种方式。 方式一 这是我的 html 文件在 ass
我正在触发一个 DatePickerDialog,在 api 22 (Android 5.1) 之前它一直在工作并显示良好,我在上面设置混合和最大日期(最小 = 当前日期,最大 = 从当前日期开始的
我有一个 ListView ,我在其中将标题 View 添加到该列表。一切都很好,但是当滚动列表 headerview 也随着列表移动时,所以我想避免 headerview 滚动,我的意思是当我列表到
虽然我在android上做过一些app,但我还是一头雾水。是否可以使用 SDK 4.0 中的功能,并在 android 2.1 或更低版本上运行该应用程序? 我尝试了你们提到的方法,但出现错误 - F
您好,我正在开发小型 android 应用程序,我想在其中显示带有一些元素的简单 gridview。它工作正常。唯一的问题是即使有空间,它也总是只显示两列。它平均将屏幕分成 2 列并仅显示两个元素。如
我正在使用 Android 2.3.3 API 构建一个应用程序。我需要识别方向的变化并执行一些操作。所以我在 Android Manifest 中添加了以下内容, android:configCha
我正在尝试在“点击”包含特定 MIME 类型的 nfc 标签时开始一项 Activity 。我制作了一个 mime 类型为“text/plain”的标签,并将其添加到 list 中:
我可以将一些数据保存到文件中 val byteArrayOutputStream = ByteArrayOutputStream() byteArrayOutputStream.wri
我正在尝试解析一个包含复杂阿拉伯字母的 XML 文件.. 当我在 android 2.3.7 上测试时,并不是所有的阿拉伯字母都被支持,仍然有一些复杂的显示为方 block .. 但是在 androi
我需要编写一个方法来确定设备是平板电脑还是手机。我不需要根据这个显示不同的用户界面。我只需要有关设备的信息,以便将来我可以将其发送到指标。 在互联网上,我找到了很多方法来确定设备是否是平板电脑。我已经
我正在玩文字转语音,让我的测试应用程序更有趣。它适用于模拟器,但不适用于我的手机,因为我的默认语言环境不是英语。 但是,文本是英文的,所以 tts 当然应该使用英文。据我所知,我可以实现一个自动安装,
我正在使用 MVVM 和整洁的架构编写应用程序。在其中一个屏幕上,我需要使用 pagination 实现 RecyclerView。我将使用库 Paging3。 Android 开发者推荐在存储库层使
这个问题在这里已经有了答案: I lost my .keystore file? (12 个回答) 4年前关闭。 当我以前的操作系统损坏并安装新的(7 月 3 日)时,以前的 android_key_
我正在使用 MVVM 和整洁的架构编写应用程序。在其中一个屏幕上,我需要使用 pagination 实现 RecyclerView。我将使用库 Paging3。 Android 开发者推荐在存储库层使
在我的 v27\style.xml 中,我有以下代码来设置白色导航栏: @color/colorPrimary @color/colorPrimaryDark @color/
我想通过发送电子邮件 startActivity(Intent.createChooser(new Intent(android.content.Intent.ACTION_SEND))) 我知道要将
我实现了一个自定义 ListView ,它看起来像 Twitter 时间线。 adapter = new MyClickableListAdapter(this, R.layout.timeline,
我有一个显示启动画面的自定义对话框; mSplashDialog = new Dialog(MyActivity.this,R.layout.splash); mSplashDialog.setCon
我是一名优秀的程序员,十分优秀!