Пользовательский ListView + Xmlparser + Actionbar 4.2 Шерлок Searchview
У меня есть собственный ListView, который получает baseadapter, передавая данные XML ArrayList.
Я хочу выполнить поиск с помощью ActionbarSherlock 4.2 Searchview, поэтому после того, как пользователь поместит текст и нажмет кнопку, он надувает новый макет с результатами.
Какое лучшее руководство?
- Должен ли я реализовать Filterable на BaseAdapter? И если да, то как я могу получить tostring() из Sherlock Searchview вместо filterText.addTextChangedListener(filterTextWatcher);?
- Должен ли я создать новый BaseAdapter для загрузки только подсказки просмотра списка при поиске? И если да, у вас есть образец или учебник для ссылки?
- Или как-то иначе?
Пользовательский ListView:
public class CustomizedListView extends ListFragment {
// All static variables
static final String URL = "http://api.androidhive.info/music/music.xml";
// XML node keys
static final String KEY_SONG = "song"; // parent node
static final String KEY_ID = "id";
static final String KEY_TITLE = "title";
static final String KEY_ARTIST = "artist";
static final String KEY_DURATION = "duration";
static final String KEY_THUMB_URL = "thumb_url";
ListView list;
LazyAdapter adapter;
@Override
public void onActivityCreated(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onActivityCreated(savedInstanceState);
ArrayList<HashMap<String, String>> songsList = new ArrayList<HashMap<String, String>>();
XMLParser parser = new XMLParser();
String xml = parser.getXmlFromUrl(URL); // getting XML from URL
Document doc = parser.getDomElement(xml); // getting DOM element
NodeList nl = doc.getElementsByTagName(KEY_SONG);
// looping through all song nodes <song>
for (int i = 0; i < nl.getLength(); i++) {
// creating new HashMap
HashMap<String, String> map = new HashMap<String, String>();
Element e = (Element) nl.item(i);
// adding each child node to HashMap key => value
map.put(KEY_ID, parser.getValue(e, KEY_ID));
map.put(KEY_TITLE, parser.getValue(e, KEY_TITLE));
map.put(KEY_ARTIST, parser.getValue(e, KEY_ARTIST));
map.put(KEY_DURATION, parser.getValue(e, KEY_DURATION));
map.put(KEY_THUMB_URL, parser.getValue(e, KEY_THUMB_URL));
// adding HashList to ArrayList
songsList.add(map);
}
// Getting adapter by passing xml data ArrayList
adapter = new LazyAdapter(getActivity(), songsList);
setListAdapter(adapter);
}
}
BaseAdapter:
public class LazyAdapter extends BaseAdapter {
private FragmentActivity activity;
private ArrayList<HashMap<String, String>> data;
private static LayoutInflater inflater=null;
public ImageLoader imageLoader;
public LazyAdapter(FragmentActivity fragmentActivity, ArrayList<HashMap<String, String>> d) {
activity = fragmentActivity;
data=d;
inflater = (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
imageLoader=new ImageLoader(activity.getApplicationContext());
}
public int getCount() {
return data.size();
}
public Object getItem(int position) {
return position;
}
public long getItemId(int position) {
return position;
}
public View getView(int position, View convertView, ViewGroup parent) {
View vi=convertView;
if(convertView==null)
vi = inflater.inflate(R.layout.list_row, null);
TextView title = (TextView)vi.findViewById(R.id.title); // title
TextView artist = (TextView)vi.findViewById(R.id.artist); // artist name
TextView duration = (TextView)vi.findViewById(R.id.duration); // duration
ImageView thumb_image=(ImageView)vi.findViewById(R.id.list_image); // thumb image
HashMap<String, String> song = new HashMap<String, String>();
song = data.get(position);
// Setting all values in listview
title.setText(song.get(CustomizedListView.KEY_TITLE));
artist.setText(song.get(CustomizedListView.KEY_ARTIST));
duration.setText(song.get(CustomizedListView.KEY_DURATION));
imageLoader.DisplayImage(song.get(CustomizedListView.KEY_THUMB_URL), thumb_image);
return vi;
}
}
Шерлок ActionBar SearchView:
public abstract class BaseSampleActivity extends SherlockFragmentActivity {
private static final Random RANDOM = new Random();
TestFragmentAdapter mAdapter;
ViewPager mPager;
PageIndicator mIndicator;
@Override
public boolean onCreateOptionsMenu(Menu menu) {
super.onCreateOptionsMenu(menu);
SearchView searchView = new SearchView(getSupportActionBar().getThemedContext());
searchView.setQueryHint("Procure pelo nome");
searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
@Override
public boolean onQueryTextSubmit(String query) {
// TODO Auto-generated method stub
Intent search = new Intent(getApplicationContext(), SearchableActivity.class);
search.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
search.putExtra(query, query);
startActivityForResult(search, 0);
return true;
}
@Override
public boolean onQueryTextChange(String newText) {
// TODO Auto-generated method stub
return false;
}
});
menu.add("Search")
.setIcon(R.drawable.ic_search_inverse)
.setActionView(searchView)
.setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM | MenuItem.SHOW_AS_ACTION_COLLAPSE_ACTION_VIEW);
return true;
}
Новая активность:
public class SearchableActivity extends SherlockListActivity{
// All static variables
static final String URL = "http://api.androidhive.info/music/music.xml";
// XML node keys
static final String KEY_SONG = "song"; // parent node
static final String KEY_ID = "id";
static final String KEY_TITLE = "title";
static final String KEY_ARTIST = "artist";
static final String KEY_DURATION = "duration";
static final String KEY_THUMB_URL = "thumb_url";
private String message;
private ListView list;
private LazyAdapterS adapter;
protected void onCreate(Bundle savedInstanceState, String query) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
message = getIntent().getExtras().getString(query);
setContentView(R.layout.main_list_row);
ArrayList<HashMap<String, String>> songsList = new ArrayList<HashMap<String, String>>();
XMLParser parser = new XMLParser();
String xml = parser.getXmlFromUrl(URL); // getting XML from URL
Document doc = parser.getDomElement(xml); // getting DOM element
NodeList nl = doc.getElementsByTagName(KEY_SONG);
// looping through all song nodes <song>
for (int i = 0; i < nl.getLength(); i++) {
// creating new HashMap
HashMap<String, String> map = new HashMap<String, String>();
Element e = (Element) nl.item(i);
// adding each child node to HashMap key => value
map.put(KEY_ID, parser.getValue(e, KEY_ID));
map.put(KEY_TITLE, parser.getValue(e, KEY_TITLE));
map.put(KEY_ARTIST, parser.getValue(e, KEY_ARTIST));
map.put(KEY_DURATION, parser.getValue(e, KEY_DURATION));
map.put(KEY_THUMB_URL, parser.getValue(e, KEY_THUMB_URL));
// adding HashList to ArrayList
songsList.add(map);
}
list=(ListView)findViewById(R.id.list);
// Getting adapter by passing xml data ArrayList
adapter = new LazyAdapterS(this, songsList);
list.setAdapter(adapter);
((Filterable) list.getAdapter()).Filter(message);// Error The method Filter(String) is undefined for the type Filterable
}
}
ОБНОВЛЕНИЕ: я поместил setOnQueryTextListener как предложено @Luksprog. ОБНОВЛЕНИЕ: Я создал новое действие, чтобы показать результаты поиска, но все еще получил ошибку в фильтре, я не знаю, как с этим справиться.
Спасибо,
1 ответ
Должен ли я реализовать Filterable на BaseAdapter?
Если вы хотите использовать стандартный механизм фильтрации для адаптера (что делает любой адаптер Android SDK), да, вам нужно реализовать Filterable
интерфейс и обеспечить правильную реализацию для его двух методов.
И если да, то как я могу получить tostring() из Sherlock Searchview вместо filterText.addTextChangedListener(filterTextWatcher);?
Посмотри на OnQueryTextListener
:
public boolean onQueryTextSubmit(String query) {
((Filterable) listview.getAdapter()).filter(query);
return true;
}
Должен ли я создать новый BaseAdapter для загрузки только подсказки просмотра списка при поиске? И если да, у вас есть образец или учебник для ссылки?
Если вы имеете в виду, снова проанализируете xml(используя только то, что соответствует строке запроса) и создайте новый адаптер для отображения результатов, нет, потому что вы будете выполнять много ненужной работы (особенно все сетевые вызовы, которые вы должны действительно попробовать двигаться в фоновом потоке).
Если вы хотите показать результаты в новом действии, тогда поставьте запрос String
в Intent
это запускает действие результатов и выполняет / повторяет фильтрацию там.
Или как-то иначе?
Смотрите пункт первый.