Как скрыть один элемент в 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.