Как скрыть один элемент в Android Spinner
Я ищу способ скрыть один элемент в виджере Android spinner. Это позволит вам моделировать счетчик без выбранных элементов и гарантирует, что обратный вызов onItemSelected() всегда вызывается для каждого выбранного элемента (если скрытый элемент является "текущим" ). Обычно в счетчике всегда есть один элемент, который не генерирует обратный вызов, а именно текущий.
В stackoverflow имеется некоторый код для отключения элементов (серого), но не для того, чтобы скрыть объекты полностью, как если бы они не существовали.
После долгих экспериментов я придумал несколько хакерских решений, которые работают на разных старых и новых платформах Android. У этого есть некоторые незначительные косметические недостатки, которые трудно заметить. Я все равно хотел бы услышать о более официальном решении, кроме "не делай этого с помощью счетчика".
Это всегда скрывает первый элемент в счетчике, но его можно довольно легко расширить, чтобы скрыть произвольный элемент или более одного элемента. Добавьте фиктивный элемент, содержащий пустую строку в начале списка элементов-счетчиков. Вы можете захотеть установить текущий выбор счетчика на элемент 0 перед открытием диалога счетчика, который будет имитировать невыбранный счетчик.
Пример установки Spinner с переопределением метода ArrayAdapter:
List<String> list = new ArrayList<String>();
list.add(""); // Initial dummy entry
list.add("string1");
list.add("string2");
list.add("string3");
// Populate the spinner using a customized ArrayAdapter that hides the first (dummy) entry
ArrayAdapter<String> dataAdapter = new ArrayAdapter<String>(this, android.R.layout.simple_spinner_item, list) {
@Override
public View getDropDownView(int position, View convertView, ViewGroup parent)
{
View v = null;
// If this is the initial dummy entry, make it hidden
if (position == 0) {
TextView tv = new TextView(getContext());
tv.setHeight(0);
tv.setVisibility(View.GONE);
v = tv;
}
else {
// Pass convertView as null to prevent reuse of special case views
v = super.getDropDownView(position, null, parent);
}
// Hide scroll bar because it appears sometimes unnecessarily, this does not prevent scrolling
parent.setVerticalScrollBarEnabled(false);
return v;
}
};
dataAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
mySpinner.setAdapter(dataAdapter);
Ответы
Ответ 1
Чтобы скрыть произвольный элемент или более одного элемента, я думаю, что вы можете реализовать свой собственный адаптер и установить индекс (или список индексов массива), который вы хотите скрыть.
public class CustomAdapter extends ArrayAdapter<String> {
private int hidingItemIndex;
public CustomAdapter(Context context, int textViewResourceId, String[] objects, int hidingItemIndex) {
super(context, textViewResourceId, objects);
this.hidingItemIndex = hidingItemIndex;
}
@Override
public View getDropDownView(int position, View convertView, ViewGroup parent) {
View v = null;
if (position == hidingItemIndex) {
TextView tv = new TextView(getContext());
tv.setVisibility(View.GONE);
v = tv;
} else {
v = super.getDropDownView(position, null, parent);
}
return v;
}
}
И используйте свой собственный адаптер при создании списка элементов.
List<String> list = new ArrayList<String>();
list.add(""); // Initial dummy entry
list.add("string1");
list.add("string2");
list.add("string3");
int hidingItemIndex = 0;
CustomAdapter dataAdapter = new CustomAdapter(this, android.R.layout.simple_spinner_item, list, hidingItemIndex);
dataAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
mySpinner.setAdapter(dataAdapter);
(Я не проверял код) надеюсь, что это помогает.
Ответ 2
Легче скрыть элемент в конце списка, обрезая список.
Но сначала вы должны выбрать его, чтобы он отображался на счетчике, а затем проверить, был ли выбор изменен на один из отображаемых элементов.
List<String> list = new ArrayList<String>();
list.add("string1");
list.add("string2");
list.add("string3");
list.add("[Select one]");
final int listsize = list.size() - 1;
ArrayAdapter<String> dataAdapter = new ArrayAdapter<String>(this,android.R.layout.simple_spinner_item, list) {
@Override
public int getCount() {
return(listsize); // Truncate the list
}
};
dataAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
mySpinner.setAdapter(dataAdapter);
mySpinner.setSelection(listsize); // Hidden item to appear in the spinner
Ответ 3
Чтобы скрыть какой-либо элемент в раскрывающемся списке счетчиков, вам необходимо передать позицию элемента, которая должна быть скрыта на основании требуемых критериев.
Я достиг этого в прецеденте, скрывающем элемент, который выбран из раскрывающегося списка
public class CustomAdapter extends ArrayAdapter<String> {
private List<String> dates;
private int hideItemPostion;
public CustomAdapter (Context context, int resource, List<String> dates) {
super(context, resource,dates);
this.dates=dates;
}
public void setItemToHide(int itemToHide)
{
this.hideItemPostion =itemToHide;
}
@Override
public View getDropDownView(int position, View convertView, ViewGroup parent) {
View v = null;
if (position == hideItemPostion) {
TextView tv = new TextView(getContext());
tv.setVisibility(View.GONE);
tv.setHeight(0);
v = tv;
v.setVisibility(View.GONE);
}
else
v = super.getDropDownView(position, null, parent);
return v;
}}
И настройка адаптера примерно такая
final CustomAdapter dataAdapter = new CustomAdapter(this,R.layout.spinner_item,dates);
dataAdapter.setDropDownViewResource(R.layout.spinner_dropdown_item);
spinner.setAdapter(dataAdapter);
dataAdapter.setItemToHide(0);
При выборе некоторых элементов в раскрывающемся меню также необходимо изменить положение
spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
@Override
public void onItemSelected(AdapterView<?> adapterView, View view, final int i, long l) {
dataAdapter.notifyDataSetChanged();
mEPGDateSelector.setSelection(i);
dataAdapter.setItemToHide(i);}
@Override
public void onNothingSelected(AdapterView<?> adapterView) {
}
});
Ответ 4
Просто для интереса, я сделал решение использовать "Подскажите" в качестве подсказки. Этот код создан для Xamarin.Android
, но он может быть отлично портирован на Java за 10 минут. Используйте его как простой ArrayAdapter
, не добавляя в исходный массив индексированные или индексированные по индексу числа. Он также устанавливает SpinnerGeolocation.SelectedItemId
в -1, когда ничего не выбрано (hint
- текущий элемент).
public class ArrayAdapterWithHint<T>: ArrayAdapter<T>
{
protected bool HintIsSet = false;
protected int HintResource = 0;
public ArrayAdapterWithHint(Context context, int textViewResourceId,
T[] objects)
: base(context, textViewResourceId, objects)
{
}
public ArrayAdapterWithHint(Context context, int hintResource,
int textViewResourceId, T[] objects)
: base(context, textViewResourceId, objects)
{
HintResource = hintResource;
}
public ArrayAdapterWithHint(Context context, int textViewResourceId,
IList<T> objects)
: base(context, textViewResourceId, objects)
{
}
public ArrayAdapterWithHint(Context context, int hintResource,
int textViewResourceId, IList<T> objects)
: base(context, textViewResourceId, objects)
{
HintResource = hintResource;
}
public override View GetDropDownView(int position, View convertView,
ViewGroup parent)
{
if (HintIsSet)
return base.GetDropDownView(position + 1,
convertView, parent);
return base.GetDropDownView(position, convertView, parent);
}
public override View GetView(int position, View convertView,
ViewGroup parent)
{
if (!HintIsSet && parent is Spinner &&
!string.IsNullOrWhiteSpace((parent as Spinner).Prompt))
{
Insert((parent as Spinner).Prompt, 0);
HintIsSet = true;
(parent as Spinner).SetSelection(base.Count - 1);
}
if (HintIsSet && position >= base.Count - 1)
{
View hintView = base.GetView(0, convertView, parent);
if (hintView is TextView)
(hintView as TextView).SetTextAppearance(
Context, HintResource);
return hintView;
}
if (HintIsSet && position < base.Count - 1)
return base.GetView(position + 1, convertView, parent);
return base.GetView(position, convertView, parent);
}
public override long GetItemId(int position)
{
if (HintIsSet)
{
if (position >= base.Count - 1)
return -1;
return position;
}
return base.GetItemId(position);
}
public override int Count
{
get
{
return base.Count > 0 && HintIsSet ? base.Count - 1 : base.Count;
}
}
}
Ответ 5
Я нашел это решение, которое решило мою проблему.
final Spinner mySpinner = (Spinner)findViewById(R.id.spinner_triptype);
final ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,R.layout.spinner_item, R.id.weekofday, triptype_initial);
final ArrayAdapter<String> adapter_temp = new ArrayAdapter<String>
(this,R.layout.spinner_item, R.id.weekofday, triptype_array);
mySpinner.setAdapter(adapter);
mySpinner.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
// display your error popup here
if(flag_spinner_isFirst){
mySpinner.setAdapter(adapter_temp);
flag_spinner_isFirst = false;
}
v.onTouchEvent(event);
return true;
}
});
Ответ 6
Я думаю, что лучше разместить проверку в списке массивов, а не на Spinner, потому что, как только элемент будет отфильтрован, его можно будет добавить в Spinner
Ответ 7
Еще один подход, который лучше всего подходит для меня, - это вернуть новый пустой объект. Это значительно чистый подход, поскольку вы не играете с элементами массива.
Создайте класс адаптера, расширяющий ArrayAdapter
внутри вашего метода
public View getView(int position, View convertView, ViewGroup parent) {
View row = getCustomView();
if(position==0) // put the desired check here.
{
row = new View(context);
}
}
return row;
}
Ответ 8
Это очень старый вопрос, но я нашел хороший (и, вероятно,) чистый способ не показывать первые элементы.
Вдохновленный ответом @Romich, я добавил похожую логику, чтобы пропустить первый пункт (ы).
Это эффективно скрывает произвольное количество элементов (1 по умолчанию). Код сообщает только о том, что размер отображаемых объектов короче, чем он есть на самом деле, а также изменяет индекс отображаемых элементов, поэтому мы пропускаем произвольное количество элементов.
Для простоты я исключил решение, которое в настоящее время использую, которое поддерживает скрытие списка случайных элементов, но этим легко можно управлять с помощью нескольких настроек кода.
class ArrayAdapterCustom(context: Context, textViewResourceId: Int, vararg objects: String)
: ArrayAdapter<String>(context, textViewResourceId, objects) {
//Can skip first n items (skip 1 as default)
var hideFirstItemsCount = 1
override fun getCount(): Int {
return super.getCount() - hideFirstItemsCount
}
override fun getDropDownView(position: Int, convertView: View?, parent: ViewGroup): View {
return super.getDropDownView(position + hideFirstItemsCount, convertView, parent)
}
}
Ответ 9
почему вы хотите скрыть элемент. Просто покажите только те, которые требуются, и он предлагает быстрый вариант, чтобы вы могли указать пользователя о описании spinner.
И я согласен с Akhilesh, что лучше фильтровать этот элемент на этапе ArrayList.