Как установить выбранный элемент Spinner по значению, а не по положению?

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

Я имел в виду что-то вроде этого, но у Adapter нет метода indexOf , поэтому я застрял.

 void setSpinner(String value) { int pos = getSpinnerField().getAdapter().indexOf(value); getSpinnerField().setSelection(pos); } 

Solutions Collecting From Web of "Как установить выбранный элемент Spinner по значению, а не по положению?"

Предположим, что ваш Spinner назван mSpinner , и он содержит как один из его вариантов: «некоторое значение».

Чтобы найти и сравнить позицию «некоторое значение» в Spinner, используйте это:

 String compareValue = "some value"; ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(this, R.array.select_state, android.R.layout.simple_spinner_item); adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item); mSpinner.setAdapter(adapter); if (!compareValue.equals(null)) { int spinnerPosition = adapter.getPosition(compareValue); mSpinner.setSelection(spinnerPosition); } 

Простым способом установки счетчика на основе значения является

 mySpinner.setSelection(getIndex(mySpinner, myValue)); //private method of your class private int getIndex(Spinner spinner, String myString) { int index = 0; for (int i=0;i<spinner.getCount();i++){ if (spinner.getItemAtPosition(i).toString().equalsIgnoreCase(myString)){ index = i; break; } } return index; } 

Путь к сложному коду уже существует, это намного проще.

Я держу отдельный ArrayList всех предметов в моих Спиннерах. Таким образом, я могу сделать indexOf в ArrayList, а затем использовать это значение для установки выделения в Spinner.

Основываясь на ответе Merrill , я придумал это однолинейное решение … это не очень красиво, но вы можете обвинить любого, кто поддерживает код для Spinner для пренебрежения включением функции, которая делает это для этого.

 mySpinner.setSelection(((ArrayAdapter<String>)mySpinner.getAdapter()).getPosition(myString)); 

Вы получите предупреждение о том, что приведение в ArrayAdapter<String> не проверено … действительно, вы могли бы просто использовать ArrayAdapter как это ArrayAdapter Merrill, но это просто обменяет одно предупреждение на другое.

Основываясь на ответе Merrill, вот как это сделать с CursorAdapter

 CursorAdapter myAdapter = (CursorAdapter) spinner_listino.getAdapter(); //cast for(int i = 0; i < myAdapter.getCount(); i++) { if (myAdapter.getItemId(i) == ordine.getListino() ) { this.spinner_listino.setSelection(i); break; } } 

Если вам нужен метод indexOf для любого старого адаптера (и вы не знаете, как он работает), вы можете использовать это:

 private int indexOf(final Adapter adapter, Object value) { for (int index = 0, count = adapter.getCount(); index < count; ++index) { if (adapter.getItem(index).equals(value)) { return index; } } return -1; } 

Если вы используете строковый массив, это лучший способ:

 int selectionPosition= adapter.getPosition("YOUR_VALUE"); spinner.setSelection(selectionPosition); 

Вы также можете использовать это,

 String[] baths = getResources().getStringArray(R.array.array_baths); mSpnBaths.setSelection(Arrays.asList(baths).indexOf(value_here)); 

Используйте следующую строку, чтобы выбрать значение:

 mSpinner.setSelection(yourList.indexOf("value")); 

Вот как это сделать, если вы используете SimpleCursorAdapter (где columnName – это имя столбца db, который вы использовали для заполнения вашего spinner ):

 private int getIndex(Spinner spinner, String columnName, String searchString) { //Log.d(LOG_TAG, "getIndex(" + searchString + ")"); if (searchString == null || spinner.getCount() == 0) { return -1; // Not found } else { Cursor cursor = (Cursor)spinner.getItemAtPosition(0); for (int i = 0; i < spinner.getCount(); i++) { cursor.moveToPosition(i); String itemText = cursor.getString(cursor.getColumnIndex(columnName)); if (itemText.equals(searchString)) { return i; } } return -1; // Not found } } 

(Возможно, вам также потребуется закрыть курсор, в зависимости от того, используете ли вы загрузчик.)

Также (уточнение ответа Ахиля ) это соответствующий способ сделать это, если вы заполняете Spinner из массива:

 private int getIndex(Spinner spinner, String searchString) { if (searchString == null || spinner.getCount() == 0) { return -1; // Not found } else { for (int i = 0; i < spinner.getCount(); i++) { if (spinner.getItemAtPosition(i).toString().equals(searchString)) { return i; // Found! } } return -1; // Not found } }; 

Это мой простой способ получить индекс по строке.

 private int getIndexByString(Spinner spinner, String string) { int index = 0; for (int i = 0; i < spinner.getCount(); i++) { if (spinner.getItemAtPosition(i).toString().equalsIgnoreCase(string)) { index = i; break; } } return index; } 

На самом деле есть способ получить это, используя индексный поиск в AdapterArray, и все это можно сделать с отражением. Я даже сделал еще один шаг вперед, так как у меня было 10 Spinners и я хотел динамически их устанавливать из своей базы данных, а база данных хранит значение только не в тексте, поскольку Spinner фактически меняется с недели на неделю, поэтому значение – это номер моего идентификатора из базы данных.

  // Get the JSON object from db that was saved, 10 spinner values already selected by user JSONObject json = new JSONObject(string); JSONArray jsonArray = json.getJSONArray("answer"); // get the current class that Spinner is called in Class<? extends MyActivity> cls = this.getClass(); // loop through all 10 spinners and set the values with reflection for (int j=1; j< 11; j++) { JSONObject obj = jsonArray.getJSONObject(j-1); String movieid = obj.getString("id"); // spinners variable names are s1,s2,s3... Field field = cls.getDeclaredField("s"+ j); // find the actual position of value in the list int datapos = indexedExactSearch(Arrays.asList(Arrays.asList(this.data).toArray()), "value", movieid) ; // find the position in the array adapter int pos = this.adapter.getPosition(this.data[datapos]); // the position in the array adapter ((Spinner)field.get(this)).setSelection(pos); } .  // Get the JSON object from db that was saved, 10 spinner values already selected by user JSONObject json = new JSONObject(string); JSONArray jsonArray = json.getJSONArray("answer"); // get the current class that Spinner is called in Class<? extends MyActivity> cls = this.getClass(); // loop through all 10 spinners and set the values with reflection for (int j=1; j< 11; j++) { JSONObject obj = jsonArray.getJSONObject(j-1); String movieid = obj.getString("id"); // spinners variable names are s1,s2,s3... Field field = cls.getDeclaredField("s"+ j); // find the actual position of value in the list int datapos = indexedExactSearch(Arrays.asList(Arrays.asList(this.data).toArray()), "value", movieid) ; // find the position in the array adapter int pos = this.adapter.getPosition(this.data[datapos]); // the position in the array adapter ((Spinner)field.get(this)).setSelection(pos); } 

Вот индексный поиск, который вы можете использовать практически в любом списке, если поля находятся на верхнем уровне объекта.

  /** * Searches for exact match of the specified class field (key) value within the specified list. * This uses a sequential search through each object in the list until a match is found or end * of the list reached. It may be necessary to convert a list of specific objects into generics, * ie: LinkedList&ltDevice&gt needs to be passed as a List&ltObject&gt or Object[&nbsp] by using * Arrays.asList(device.toArray(&nbsp)). * * @param list - list of objects to search through * @param key - the class field containing the value * @param value - the value to search for * @return index of the list object with an exact match (-1 if not found) */ public static <T> int indexedExactSearch(List<Object> list, String key, String value) { int low = 0; int high = list.size()-1; int index = low; String val = ""; while (index <= high) { try { //Field[] c = list.get(index).getClass().getDeclaredFields(); val = cast(list.get(index).getClass().getDeclaredField(key).get(list.get(index)) , "NONE"); } catch (SecurityException e) { e.printStackTrace(); } catch (NoSuchFieldException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } if (val.equalsIgnoreCase(value)) return index; // key found index = index + 1; } return -(low + 1); // key not found return -1 } 

Метод Cast, который может быть создан для всех примитивов, является одним для string и int.

  /** * Base String cast, return the value or default * @param object - generic Object * @param defaultValue - default value to give if Object is null * @return - returns type String */ public static String cast(Object object, String defaultValue) { return (object!=null) ? object.toString() : defaultValue; } /** * Base integer cast, return the value or default * @param object - generic Object * @param defaultValue - default value to give if Object is null * @return - returns type integer */ public static int cast(Object object, int defaultValue) { return castImpl(object, defaultValue).intValue(); } /** * Base cast, return either the value or the default * @param object - generic Object * @param defaultValue - default value to give if Object is null * @return - returns type Object */ public static Object castImpl(Object object, Object defaultValue) { return object!=null ? object : defaultValue; } 

Вот мое решение

 List<Country> list = CountryBO.GetCountries(0); CountriesAdapter dataAdapter = new CountriesAdapter(this,list); dataAdapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item); spnCountries.setAdapter(dataAdapter); spnCountries.setSelection(dataAdapter.getItemIndexById(userProfile.GetCountryId())); 

И getItemIndexById ниже

 public int getItemIndexById(String id) { for (Country item : this.items) { if(item.GetId().toString().equals(id.toString())){ return this.items.indexOf(item); } } return 0; } 

Надеюсь эта помощь!

Я использую пользовательский адаптер, для этого достаточно этого кода:

 yourSpinner.setSelection(arrayAdapter.getPosition("Your Desired Text")); 

Итак, ваш фрагмент кода будет выглядеть так:

 void setSpinner(String value) { yourSpinner.setSelection(arrayAdapter.getPosition(value)); } 

Чтобы приложение запомнило последние выбранные значения счетчика, вы можете использовать код ниже:

  1. Ниже кода считывается значение счетчика и соответственно устанавливается позиция счетчика.

     public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); int spinnerPosition; Spinner spinner1 = (Spinner) findViewById(R.id.spinner1); ArrayAdapter<CharSequence> adapter1 = ArrayAdapter.createFromResource( this, R.array.ccy_array, android.R.layout.simple_spinner_dropdown_item); adapter1.setDropDownViewResource(android.R.layout.simple_list_item_activated_1); // Apply the adapter to the spinner spinner1.setAdapter(adapter1); // changes to remember last spinner position spinnerPosition = 0; String strpos1 = prfs.getString("SPINNER1_VALUE", ""); if (strpos1 != null || !strpos1.equals(null) || !strpos1.equals("")) { strpos1 = prfs.getString("SPINNER1_VALUE", ""); spinnerPosition = adapter1.getPosition(strpos1); spinner1.setSelection(spinnerPosition); spinnerPosition = 0; } 
  2. И поставьте ниже код, в котором вы знаете, какие последние значения прядильщика присутствуют, или где-то еще по мере необходимости. Этот фрагмент кода в основном записывает значение spinner в SharedPreferences.

      Spinner spinner1 = (Spinner) findViewById(R.id.spinner1); String spinlong1 = spinner1.getSelectedItem().toString(); SharedPreferences prfs = getSharedPreferences("WHATEVER", Context.MODE_PRIVATE); SharedPreferences.Editor editor = prfs.edit(); editor.putString("SPINNER1_VALUE", spinlong1); editor.commit(); 

У меня была такая же проблема при попытке выбрать правильный элемент в spinner, заполненном с помощью cursorLoader. Я получил идентификатор элемента, который я хотел выбрать первым из таблицы 1, а затем использовал CursorLoader для заполнения счетчика. В onLoadFinished я прошел через курсор, заполняющий адаптер счетчика, пока не нашел элемент, который соответствовал идентификатору, который у меня уже был. Затем назначьте номер строки курсора на выбранную позицию счетчика. Было бы неплохо иметь аналогичную функцию для передачи идентификатора значения, которое вы хотите выбрать в счетчике, при заполнении сведений о форме, содержащей сохраненные результаты spinner.

 @Override public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) { adapter.swapCursor(cursor); cursor.moveToFirst(); int row_count = 0; int spinner_row = 0; while (spinner_row < 0 || row_count < cursor.getCount()){ // loop until end of cursor or the // ID is found int cursorItemID = bCursor.getInt(cursor.getColumnIndexOrThrow(someTable.COLUMN_ID)); if (knownID==cursorItemID){ spinner_row = row_count; //set the spinner row value to the same value as the cursor row } cursor.moveToNext(); row_count++; } } spinner.setSelection(spinner_row ); //set the selected item in the spinner } 

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

Если вы установите значения в ArrayList с помощью String.format , вы ДОЛЖНЫ получить позицию значения, используя ту же String.format структуру String.format .

Пример:

 ArrayList<String> myList = new ArrayList<>(); myList.add(String.format(Locale.getDefault() ,"%d", 30)); myList.add(String.format(Locale.getDefault(), "%d", 50)); myList.add(String.format(Locale.getDefault(), "%d", 70)); myList.add(String.format(Locale.getDefault(), "%d", 100)); 

Вы должны получить нужную величину следующим образом:

 myList.setSelection(myAdapter.getPosition(String.format(Locale.getDefault(), "%d", 70))); 

В противном случае вы получите -1 , элемент не найден!

Я использовал Locale.getDefault() из-за арабского языка .

Надеюсь, это будет полезно для вас.

Вот мое, надеюсь, полное решение. У меня есть следующее перечисление:

 public enum HTTPMethod {GET, HEAD} 

Используется в следующем классе

 public class WebAddressRecord { ... public HTTPMethod AccessMethod = HTTPMethod.HEAD; ... 

Код для установки счетчика по элементу перечисления HTTPMethod:

  Spinner mySpinner = (Spinner) findViewById(R.id.spinnerHttpmethod); ArrayAdapter<HTTPMethod> adapter = new ArrayAdapter<HTTPMethod>(this, android.R.layout.simple_spinner_item, HTTPMethod.values()); mySpinner.setAdapter(adapter); int selectionPosition= adapter.getPosition(webAddressRecord.AccessMethod); mySpinner.setSelection(selectionPosition); 

Где R.id.spinnerHttpmethod определяется в макете-файле, а android.R.layout.simple_spinner_item доставляется андроид-студией.

Вы должны передать свой пользовательский адаптер с позицией, подобной REPEAT [position]. И он работает правильно.