Создание setError () для Spinner - PullRequest
       2

Создание setError () для Spinner

44 голосов
/ 20 сентября 2010

Как создать функцию setError() (аналогичную функции TextView/EditText) для Spinner?Следующее не работает:

Я попытался расширить класс Spinner и в конструкторе:

ArrayAdapter<String> aa = new ArrayAdapter<String>(getContext(),
                    android.R.layout.simple_spinner_item, android.R.id.text1,
                    items);
            aa.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
            setAdapter(aa);
             tv = (TextView) findViewById(android.R.id.text1);
            // types_layout_list_tv

            ctv = (CheckedTextView) aa.getDropDownView(1, null, null);
            tv2 = (TextView) aa.getView(1, null, null);

setError метод:

    public void setError(String str) {
        if (tv != null)
            tv.setError(str);
        if(tv2!=null)
            tv2.setError(str);
        if (ctv != null)
            ctv.setError(str);
    }

Ответы [ 9 ]

40 голосов
/ 08 декабря 2014

Похоже на решение @ Gábor, но мне не нужно было создавать собственный адаптер. Я просто вызываю следующий код в моей функции проверки (т.е. при нажатии кнопки подтверждения)

        TextView errorText = (TextView)mySpinner.getSelectedView();                  
        errorText.setError("anything here, just to add the icon");
        errorText.setTextColor(Color.RED);//just to highlight that this is an error
        errorText.setText("my actual error text");//changes the selected item text to this
20 голосов
/ 24 марта 2014

У меня есть решение, которое не требует создания дополнительного поля редактирования, но вам, как обычно, нужно ваше собственное SpinnerAdapter.

Убедитесь, что у вас есть хотя бы один TextView в используемом макетев getView() вашего адаптера (обычно у вас это так или иначе).

Добавьте следующую функцию к вашему адаптеру (измените name на идентификатор вашего TextView):

public void setError(View v, CharSequence s) {
  TextView name = (TextView) v.findViewById(R.id.name);
  name.setError(s);
}

Вызовите setError() из своего кода следующим образом:

YourAdapter adapter = (YourAdapter)spinner.getAdapter();
View view = spinner.getSelectedView();
adapter.setError(view, getActivity().getString(R.string.error_message));

По сути, как и для любого другого элемента управления, только то, что вы вызываете его на своем адаптере, и вы также должны предоставить представление.

Это отобразит значок ошибки на счетчике, как в случае с другими элементами управления.

11 голосов
/ 30 апреля 2015

Использование скрытого TextView для отображения всплывающего сообщения

Это решение включает в себя добавление дополнительного скрытого текстового поля чуть ниже вращающегося элемента в правильном положении, чтобы можно было отобразить диалоговое окно ошибки TextView, в то же время также используя TextView, установленный в XML макета счетчика, чтобы красный (!) Значок отображался отображается. Таким образом, в действительности используются два текстовых представления - одно для значка, а другое (скрытое) для разрешения диалога об ошибке.

Вот как это выглядит, когда нет ошибки (используйте SetError(null)):

Spinner in valid state

Вот как это выглядит при возникновении ошибки (используйте SetError("my error text, ideally from a resource!")):

Spinner in invalid state

Вот выдержка из макета блесны XML. Существует RelativeLayout, который используется для того, чтобы TextView был как можно ближе к счетчику, и имеет достаточно paddingRight, чтобы стрелка в диалоговом окне сообщения была выровнена под красным значком ошибки (!). Скрытый (поддельный) TextView расположен относительно Spinner.

    <RelativeLayout
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:gravity="top|left"
        >

        <Spinner
            android:id="@+id/spnMySpinner"
            android:layout_width="400dp"
            android:layout_height="wrap_content"
            android:layout_alignParentTop="true"
            android:dropDownSelector="@drawable/selector_listview"
            android:background="@android:drawable/btn_dropdown"
            android:paddingBottom="0dp"
            android:layout_marginBottom="0dp"
            />

        <!-- Fake TextView to use to set in an error state to allow an error to be shown for the TextView -->
        <android.widget.TextView
            android:id="@+id/tvInvisibleError"
            android:layout_width="0dp"
            android:layout_height="0dp"
            android:layout_alignRight="@+id/spnMySpinner"
            android:layout_alignBottom="@+id/spnMySpinner"
            android:layout_marginTop="0dp"
            android:paddingTop="0dp"
            android:paddingRight="50dp"
            android:focusable="true"
            android:focusableInTouchMode="true"
            />

    </RelativeLayout>

Примечание: @drawable/selector_listview определено за пределами данного решения. См. Пример здесь о том, как заставить это работать, поскольку это не по теме для этого ответа.

Вот код, чтобы заставить его работать. Просто вызовите SetError(errMsg) либо с null, чтобы очистить ошибку, либо с текстом, чтобы установить ее в состояние ошибки.

/**
 * When a <code>errorMessage</code> is specified, pops up an error window with the message
 * text, and creates an error icon in the secondary unit spinner. Error cleared through passing
 * in a null string.
 * @param errorMessage Error message to display, or null to clear.
 */
public void SetError(String errorMessage)
{
    View view = spnMySpinner.getSelectedView();

    // Set TextView in Secondary Unit spinner to be in error so that red (!) icon
    // appears, and then shake control if in error
    TextView tvListItem = (TextView)view;

    // Set fake TextView to be in error so that the error message appears
    TextView tvInvisibleError = (TextView)findViewById(R.id.tvInvisibleError);

    // Shake and set error if in error state, otherwise clear error
    if(errorMessage != null)
    {
        tvListItem.setError(errorMessage);
        tvListItem.requestFocus();

        // Shake the spinner to highlight that current selection 
        // is invalid -- SEE COMMENT BELOW
        Animation shake = AnimationUtils.loadAnimation(this, R.anim.shake);
        spnMySpinner.startAnimation(shake);

        tvInvisibleError.requestFocus();
        tvInvisibleError.setError(errorMessage);
    }
    else
    {
        tvListItem.setError(null);
        tvInvisibleError.setError(null);
    }
}

В приведенной выше функции SetError есть некоторый дополнительный код, который вызывает дрожание текста в Spinner при установке ошибки. Это не обязательно, чтобы решение работало, но это хорошее дополнение. См. Здесь для вдохновения для этого подхода.

Спасибо @ Gábor за его решение , которое использует TextView в XML макета элемента Spinner. Код View view = spnMySpinner.getSelectedView(); (основанный на решении @ Gábor) необходим, поскольку он получает отображаемый в данный момент TextView, а не использует findViewById, который просто получит первый TextView в списке (на основе предоставленного идентификатора ресурса), и, следовательно, не будет работать (для отображения красного (!) значка), если самый первый элемент в списке не выбран.

6 голосов
/ 10 февраля 2015

Это можно сделать без использования пользовательского макета или адаптера.

((TextView)spinner.getChildAt(0)).setError("Message");

Единственным недостатком этого подхода является то, что небольшое всплывающее окно с текстом ошибки не будет отображаться при нажатии значка.

5 голосов
/ 28 января 2014

Я бы посоветовал вам поставить пустой EditText прямо за вращателем.

На xml установить, что EditText

android:enabled="false"
    android:inputType="none"

Теперь, когда вы хотите установить ошибкуна свой счетчик, просто установите эту ошибку на EditText.

Не забудьте установить EditText на invisibille / gone.Это не сработает.

Кроме того, обратите внимание, что с помощью этого метода вы можете точно определить, где вы хотите, чтобы ваша ошибка появилась.

1 голос
/ 31 декабря 2014

Спасибо Габору за ваше фантастическое решение. В поддержку вашей точки зрения мое решение таково:

Пользовательский адаптер

    public class RequiredSpinnerAdapter<T> extends ArrayAdapter<T> {
        public RequiredSpinnerAdapter(Context context, int textViewResourceId,
                                      java.util.List<T> objects) {
            super(context, textViewResourceId, objects);
        }

        int textViewId = 0;

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            View view = super.getView(position, convertView, parent);
            if (view instanceof TextView) {
                textViewId = view.getId();
            }
            return view;
        }

        public View getDropDownView(int position, View convertView, ViewGroup parent) {
            View row = super.getView(position, convertView, parent);
            return (row);
        }

        public void setError(View v, CharSequence s) {
            if(textViewId != 0){
                TextView name = (TextView) v.findViewById(textViewId);
                name.setError(s);
            }
        }
    }

Используйте адаптер для Spinner

ArrayAdapter<String> arrayAdapter = new RequiredSpinnerAdapter<String>(PropertyAdd.this, R.layout.checked, status_arr);
    marketstatus_spinner.setAdapter(arrayAdapter);
    marketstatus_spinner.setOnItemSelectedListener(new OnItemSelectedListener() {
        @Override
        public void onItemSelected(AdapterView<?> arg0, View arg1,
                                   int arg2, long arg3) {

            // Put code here
        }

        @Override
        public void onNothingSelected(AdapterView<?> arg0) {
           // Put code here
        }
    });

Проверка на валидацию

private boolean checkValidation() {
    if(marketstatus_spinner.getSelectedItem().toString().equals("")){
        RequiredSpinnerAdapter adapter = (RequiredSpinnerAdapter)marketstatus_spinner.getAdapter();
        View view = marketstatus_spinner.getSelectedView();
        adapter.setError(view, "Please select a value");

        return false;
    }
}
0 голосов
/ 16 ноября 2016

На самом деле это действительно так, вам просто нужно иметь только один TextView в вашем представлении, а затем получить выбранный вид с вашего счетчика, используя getSelectedView(), если основной вид в выбранном вами представлении равен TextView, тогдаПриведите ваш View к TextView и setError следующим образом:

((TextView) jobCategory.getSelectedView()).setError("Field Required");

Иначе, если Textview не является непосредственно MAIN View, тогда вам нужно найти его по ID и привести его снова и setErrorтаким образом:

 ((TextView) jobCategory.getSelectedView().findViewById(R.id.firstName)).setError("Field Required");
0 голосов
/ 06 марта 2013

Вы можете создать свой собственный адаптер (расширяет BaseAdapter, реализует SpinnerAdapter). Таким образом, вы можете получить доступ к TextViews, которые отображаются в счетчике. (методы getView и createViewFromResource - пример: ArrayAdapter ) Когда вы добавляете пустой элемент списка, чтобы позволить пользователю оставлять поле пустым, пока оно не станет обязательным (первый элемент в счетчике), вы можете сохранить его TextView как приватный член в адаптере. Затем, когда приходит время вызвать setError ("...") из Activity или Fragment, вы можете вызвать его на адаптере, который может передать его в пустой TextView.

@Override
public View getView(int position, View convertView, ViewGroup parent) {
    return createViewFromResource(position, convertView, parent, mTextViewId);
}

private View createViewFromResource(int position, View convertView, ViewGroup parent, int resource) {
    View view;
    TextView text;

    if (convertView == null) {
        view = inflater.inflate(resource, parent, false);
    } else {
        view = convertView;
    }

    try {
        text = (TextView) view;
    } catch (ClassCastException e) {
        Log.e(TAG, "You must supply a resource ID for a TextView", e);
        throw new IllegalStateException("MyAdapter requires the resource ID to be a TextView", e);
    }

    MyItem i = getItem(position);
    String s = (null != i) ? i.toString() : "";
    text.setText(s);

    if ("".equals(s) && null == mEmptyText) {
        this.mEmptyText = text;
    }

    return view;
}

public void setError(String errorMessage) {
    if (null != mEmptyText) {
        mEmptyText.setError(errorMessage);
    } else {
        Log.d(TAG, "mEmptyText is null");
    }
}
0 голосов
/ 08 июля 2012

Полагаю, что Спиннер - не то место, где можно поставить этот метод.В случае Spinner вы должны выбрать одно значение и чтобы значения в Spinner должны были фильтроваться на уровне вашего адаптера.Таким образом, пользователь может выбирать только те значения, которые есть в Spinner.

...