我正在开发这个应用程序,用户可以在其中输入 autoCompleteTextView 中的位置,它会根据此处描述的 Google Places Api 建议位置。
但是,我想通过仅在用户停止输入一段时间后才发送请求来限制应用程序发送的请求量。有谁知道如何做到这一点?
我正在开发这个应用程序,用户可以在其中输入 autoCompleteTextView 中的位置,它会根据此处描述的 Google Places Api 建议位置。
但是,我想通过仅在用户停止输入一段时间后才发送请求来限制应用程序发送的请求量。有谁知道如何做到这一点?
类似于这个答案,但稍微简洁一些,并且没有引入额外的状态。您也不需要阈值检查,因为performFiltering
仅在实际需要过滤时才调用。
子类AutoCompleteTextView
化似乎是唯一的方法,因为您不能覆盖/替换TextWatcher
添加的AutoCompleteTextView
.
public class DelayAutoCompleteTextView extends AutoCompleteTextView {
public DelayAutoCompleteTextView(Context context, AttributeSet attrs) {
super(context, attrs);
}
private final Handler mHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
DelayAutoCompleteTextView.super.performFiltering((CharSequence) msg.obj, msg.arg1);
}
};
@Override
protected void performFiltering(CharSequence text, int keyCode) {
mHandler.removeMessages(0);
mHandler.sendMessageDelayed(mHandler.obtainMessage(0, keyCode, 0, text), 750);
}
}
我找到了解决上述问题的方法,效果很好。我通过以下方式扩展了 AutoCompleteTextView。如果有人知道如何进一步改进此代码,请告诉我。
public class MyAutoCompleteTextView extends AutoCompleteTextView {
// initialization
int threshold;
int delay = 750;
Handler handler = new Handler();
Runnable run;
// constructor
public MyAutoCompleteTextView(Context context, AttributeSet attrs) {
super(context, attrs);
}
@Override
protected void performFiltering(final CharSequence text, final int keyCode) {
// get threshold
threshold = this.getThreshold();
// perform filter on null to hide dropdown
doFiltering(null, keyCode);
// stop execution of previous handler
handler.removeCallbacks(run);
// creation of new runnable and prevent filtering of texts which length
// does not meet threshold
run = new Runnable() {
public void run() {
if (text.length() > threshold) {
doFiltering(text, keyCode);
}
}
};
// restart handler
handler.postDelayed(run, delay);
}
// starts the actual filtering
private void doFiltering(CharSequence text, int keyCode) {
super.performFiltering(text, keyCode);
}
}
您可以使用 TextWatcher,并监控文本条目之间的时间间隔。我个人不喜欢对原生 Android 控件进行子类化的想法(不过,这当然是一种有效的方法,只是不是我喜欢的方法)。
文档:http: //developer.android.com/reference/android/text/TextWatcher.html
和一个 SO 链接: 如何在 Android 中使用 TextWatcher 类?
与上面的答案类似,我发现我需要继承 AutoCompleteTextView。然而,因为我住的地方互联网有点慢,我发现每次用户按下一个键时删除延迟的处理程序是一个问题。所以基本上如果我只在用户停止输入后 500 毫秒运行过滤器,结果可能需要几秒钟才能出现,这会让用户感到恼火。
因此,我不会每次都清除处理程序,而是让过滤每几百毫秒运行一次。我的代码不像 Jan Berkel 的那么干净。我相信你可以把它清理一下我太懒了,但是在 IMO 互联网速度慢的地区性能更好。
public class CustomCompleteView extends AutoCompleteTextView{
Handler handler=new Handler();
Runnable r;
boolean cleartogo=false;
boolean pending =false;
CharSequence btext;
int bkeyCode;
public CustomCompleteView(Context context) {
super(context);
// TODO Auto-generated constructor stub
}
public CustomCompleteView(Context context, AttributeSet attrs)
{
super(context,attrs);
}
public CustomCompleteView(Context context, AttributeSet attrs, int defStyle)
{
super(context,attrs,defStyle);
}
@Override
public void performFiltering(CharSequence text, int keyCode){
if (cleartogo){
cleartogo=false;
Log.d(MainActivity.DTAG,"Going to filter on " + text.toString());
pending=false;
super.performFiltering(btext, bkeyCode);
}
else
{
Log.d(MainActivity.DTAG,"Filtering rejected, too soon");
btext=text;
bkeyCode=keyCode;
if (!pending){
if (r==null)
r=new MyRunnable(this);
//try{handler.removeCallbacks(r);} catch (Exception ex){};
handler.postDelayed(r, 500);
pending=true;}
}
}
private class MyRunnable implements Runnable {
CustomCompleteView bc;
MyRunnable(CustomCompleteView c ) {
this.bc=c;
}
public void run() {
Log.d(MainActivity.DTAG,"Special Runnable running");
cleartogo=true;
bc.performFiltering(btext, bkeyCode);
}
}
}