android中的地理编码器自动完成 [英] Geocoder autocomplete in android

查看:20
本文介绍了android中的地理编码器自动完成的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我一直在用谷歌搜索,试图找到和我有同样问题的人,但没有运气.所以这是我的问题:

I've been googling my ass of trying to find someone that are having the same problem as me, without luck. So here's my problem:

当用户使用 Android 中的地理编码器键入地点名称时,我正在尝试实现地址的自动完成建议.我希望它的行为与使用组合框的 javascript 版本大致相同.

I'm trying to implement a autocomplete suggestion of addresses as the user types the name of a place using the geocoder in Android. I want this to behave much the same as the javascript version using a combbox.

我正在使用带有 AutoCompleteTextView 的布局和一个数组适配器,以便在用户键入时动态更新建议列表.在使用处理程序调用 geocoder.getFromLocationName 之前,从接收到 onTextChanged() 事件开始,我添加了 500 毫秒的延迟.如果用户在 500 毫秒内输入更多字母,则最后一个事件将被取消.我遇到的问题是建议几乎从不作为下拉列表中的可选项目出现在 UI 中.我得到了地址建议,但是当我将它们添加到附加到 autocomplatetextview 的适配器时,它们不会显示.

I am using a layout with an AutoCompleteTextView, and an arrayadapter to dynamically update the suggestionlist as the user types. I have added a 500ms delay from when the onTextChanged() event is received before a call to the geocoder.getFromLocationName is called using a Handler. If a user types more letter within 500ms, the last event will be cancelled. The problem I am encountering is that the suggestions almost never show up in the UI as selectables in the dropdown. I get the address suggestions, but when I add them to the adapter attached to the autocomplatetextview they simple wont show.

我正在使用 API 级别 7 的模拟器上运行它,其中包括 google api.

I'm running this on an emulator using API level 7, with google apis included.

现在一些源代码可以帮助您:布局:

Now some source code to aid you: The layout:

<LinearLayout android:id="@+id/searchInputLayout"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:padding="6dip"
    android:orientation="vertical">
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/searchMessage" />
    <EditText android:id="@+id/freetextInput" 
        android:hint="@string/searchFreetextLabel"
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" 
        android:background="@android:drawable/editbox_background" />
    <CheckBox android:id="@+id/includeVincinityCheckbox"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/includeVincinityLabel"
        android:checked="true"
        android:onClick="includeVincinityClick" />
    <AutoCompleteTextView android:id="@+id/locationInput" 
        android:hint="@string/locationInputHint"
        android:layout_width="fill_parent" 
        android:layout_height="wrap_content" />
    <Button android:id="@+id/searchButton"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/searchBtnLabel" 
        android:onClick="searchBtnClicked" />
    </LinearLayout>

我的活动的源代码(我省略了不相关的代码):

The source code of my activity (I've omitted code not relevant):

public class SearchLocationTabActivity extends Activity implements TextWatcher, OnItemSelectedListener {

private static final int MESSAGE_TEXT_CHANGED = 0;
private static final int AUTOCOMPLETE_DELAY = 500;
private static final int THRESHOLD = 3;
private String latitude, longitude;
private List<Address> autoCompleteSuggestionAddresses;
private ArrayAdapter<String> autoCompleteAdapter;
private Handler messageHandler;

/** Called when the activity is first created. */
@Override
public void onCreate(Bundle icicle) {
    super.onCreate(icicle);
    setContentView(R.layout.search);
    setDefaultKeyMode(DEFAULT_KEYS_SEARCH_LOCAL);

    messageHandler = new MyMessageHandler(this, this);
    autoCompleteAdapter = new ArrayAdapter<String>(this, android.R.layout.simple_dropdown_item_1line, new ArrayList<String>());
    autoCompleteAdapter.setNotifyOnChange(false);
    AutoCompleteTextView locationinput = (AutoCompleteTextView) findViewById(R.id.locationInput);
    locationinput.addTextChangedListener(this);
    locationinput.setOnItemSelectedListener(this);
    locationinput.setThreshold(THRESHOLD);
    locationinput.setAdapter(autoCompleteAdapter);
}

@Override
public void beforeTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) {
    messageHandler.removeMessages(MESSAGE_TEXT_CHANGED);
}

@Override
public void onTextChanged(CharSequence arg0, int arg1, int arg2, int arg3) {
String value = arg0.toString();
if (!"".equals(value) && value.length() >= THRESHOLD) {
    Message msg = Message.obtain(messageHandler, MESSAGE_TEXT_CHANGED, arg0.toString());
    messageHandler.sendMessageDelayed(msg, AUTOCOMPLETE_DELAY);
} else {
    autoCompleteAdapter.clear();
}
}

@Override
public void afterTextChanged(Editable arg0) {
}

@Override
public void onItemSelected(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
    if (arg2 < autoCompleteSuggestionAddresses.size()) {
        Address selected = autoCompleteSuggestionAddresses.get(arg2);
        latitude = Double.toString(selected.getLatitude());
        longitude = Double.toString(selected.getLongitude());
    }
}

private void notifyResult(List<Address> suggestions) {
    latitude = longitude = null;
    autoCompleteAdapter.clear();
    for (Address a : autoCompleteSuggestionAddresses) {
        autoCompleteAdapter.add(a.toString());//TODO: figure out a nice way to display this address in list
    }
    autoCompleteAdapter.notifyDataSetChanged();
}

@Override
public void onNothingSelected(AdapterView<?> arg0) {
    latitude = longitude = null;
}

private class MyMessageHandler extends Handler {

    private Context context;
    private AsyncTaskSubscriber subscriber;

    public MyMessageHandler(Context context, AsyncTaskSubscriber subscriber) {
        this.context = context;
        this.subscriber = subscriber;
    }

    @Override
    public void handleMessage(Message msg) {
        if (msg.what == MESSAGE_TEXT_CHANGED) {
            String enteredText = (String) msg.obj;

            try {
                autoCompleteSuggestionAddresses = new Geocoder(context).getFromLocationName(enteredText, 10);

                notifyResult(response);
            } catch (IOException ex) {
                Log.e(GeoCoderAsyncTask.class.getName(), "Failed to get autocomplete suggestions", ex);
            }
        }
    }
}
}

非常感谢任何帮助!

推荐答案

对于那些没有设法删除过滤的人,这是我所做的(在其他一些小的修改中,但我认为它们没有影响在过滤部分).另请注意,要单击要检测的项目之一,您需要添加 OnItemClickListener.

For those who did not managed to remove the filtering, here is what I've done (among other small modifications but I don't think they have an impact on the filtering part). Note also that for a click on one of the item to be detected, you need to add an OnItemClickListener.

autoCompleteAdapter = new ArrayAdapterNoFilter(this, android.R.layout.simple_dropdown_item_1line);

ArrayAdapterNoFilter 的灵感来自另一个答案:

Where ArrayAdapterNoFilter is inspired from this other answer:

public class ArrayAdapterNoFilter extends ArrayAdapter<String> {

    public ArrayAdapterNoFilter(Context context, int textViewResourceId) {
        super(context, textViewResourceId);
    }

    private static final NoFilter NO_FILTER = new NoFilter();

    /**
     * Override ArrayAdapter.getFilter() to return our own filtering.
     */
    @Override
    public Filter getFilter() {
        return NO_FILTER;
    }

    /**
     * Class which does not perform any filtering. Filtering is already done by
     * the web service when asking for the list, so there is no need to do any
     * more as well. This way, ArrayAdapter.mOriginalValues is not used when
     * calling e.g. ArrayAdapter.add(), but instead ArrayAdapter.mObjects is
     * updated directly and methods like getCount() return the expected result.
     */
    private static class NoFilter extends Filter {
        protected FilterResults performFiltering(CharSequence prefix) {
            return new FilterResults();
        }

        protected void publishResults(CharSequence constraint, FilterResults results) {
            // Do nothing
        }
    }
}

这篇关于android中的地理编码器自动完成的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆