195

Spinner我用 a和 a创建了一个 Android 应用程序TextView。我想在 TextView 的 Spinner 下拉列表中显示所选项目。我在方法中实现了 Spinner,onCreate所以当我运行程序时,它会在TextView(在从下拉列表中选择一个项目之前)中显示一个值。

我只想在从下拉列表中选择一个项目后在 TextView 中显示该值。我该怎么做呢?

这是我的代码:

import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemSelectedListener;
import android.widget.ArrayAdapter;
import android.widget.Spinner;
import android.widget.TextView;

public class GPACal01Activity extends Activity implements OnItemSelectedListener {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        Spinner spinner = (Spinner) findViewById(R.id.noOfSubjects);

        // Create an ArrayAdapter using the string array and a default spinner layout
        ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(this,R.array.noofsubjects_array, android.R.layout.simple_spinner_item);
        // Specify the layout to use when the list of choices appears
        adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
        // Apply the adapter to the spinner
        spinner.setAdapter(adapter);
        spinner.setOnItemSelectedListener(this);
    }

    public void onItemSelected(AdapterView<?> parent, View arg1, int pos,long id) {
        TextView textView = (TextView) findViewById(R.id.textView1);
        String str = (String) parent.getItemAtPosition(pos);
        textView.setText(str);
    }

    public void onNothingSelected(AdapterView<?> arg0) {
        // TODO Auto-generated method stub

    }
}
4

20 回答 20

202
spinner.setOnItemSelectedListener(this); // Will call onItemSelected() Listener.

所以第一次用任何整数值处理这个

示例:最初采取int check = 0;

public void onItemSelected(AdapterView<?> parent, View arg1, int pos,long id) {
   if(++check > 1) {
      TextView textView = (TextView) findViewById(R.id.textView1);
      String str = (String) parent.getItemAtPosition(pos);
      textView.setText(str);
   }
}

您可以使用布尔值执行此操作,也可以检查当前和以前的位置。看这里

于 2012-11-15T13:02:21.480 回答
168

只需在设置之前放置此行OnItemSelectedListener

spinner.setSelection(0,false)

这是有效的,因为在内部setSelection(int, boolean)调用setSelectionInt(),以便在添加侦听器时,该项目已被选中。

请注意这setSelection(int)不起作用,因为它在setNextSelectedPositionInt()内部调用。

于 2016-06-01T06:58:35.143 回答
67

从 API 级别 3 开始,您可以在带有布尔值的 Activity 上使用 onUserInteraction() 来确定用户是否正在与设备交互。

http://developer.android.com/reference/android/app/Activity.html#onUserInteraction()

@Override
public void onUserInteraction() {
     super.onUserInteraction();
     userIsInteracting = true;
}

作为 Activity 的一个字段,我有:

 private boolean userIsInteracting;

最后,我的微调器:

      mSpinnerView.setOnItemSelectedListener(new OnItemSelectedListener() {

           @Override
           public void onItemSelected(AdapterView<?> arg0, View view, int position, long arg3) {
                spinnerAdapter.setmPreviousSelectedIndex(position);
                if (userIsInteracting) {
                     updateGUI();
                }
           }

           @Override
           public void onNothingSelected(AdapterView<?> arg0) {

           }
      });

当您来回进行活动时,布尔值将重置为 false。奇迹般有效。

于 2014-07-31T23:42:09.643 回答
25

这对我有用

Spinner 在 Android 中的初始化是有问题的,有时上面的问题是通过这种模式解决的。

Spinner.setAdapter();
Spinner.setSelected(false);  // must
Spinner.setSelection(0,true);  //must
Spinner.setonItemSelectedListener(this);

在初始化微调器时,设置适配器应该是第一部分,onItemSelectedListener(this) 将是最后一部分。通过上面的模式,在微调器初始化期间不调用我的 OnItemSelected()

于 2017-06-23T07:53:30.283 回答
13

避免在初始化期间调用 spinner.setOnItemSelectedListener()

spinner.setSelection(Adapter.NO_SELECTION, true); //Add this line before setting listener
spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
    @Override
    public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {

    }

    @Override
    public void onNothingSelected(AdapterView<?> parent) {

    }
});
于 2018-11-28T09:04:38.783 回答
12

哈哈...我有同样的问题。当 initViews() 这样做时。序列是关键,侦听器是最后一个。祝你好运 !

spinner.setAdapter(adapter);
spinner.setSelection(position);
spinner.setOnItemSelectedListener(listener);
于 2015-03-14T01:26:38.857 回答
8

我的解决方案:

protected boolean inhibit_spinner = true;


@Override
        public void onItemSelected(AdapterView<?> arg0, View arg1,
                int pos, long arg3) {

            if (inhibit_spinner) {
                inhibit_spinner = false;
            }else {

            if (getDataTask != null) getDataTask.cancel(true);
            updateData();
            }

        }
于 2013-09-23T15:32:24.833 回答
6

您可以通过以下方式执行此操作:

AdapterView.OnItemSelectedListener listener = new AdapterView.OnItemSelectedListener() {
        @Override
        public void onItemSelected(AdapterView<?> adapterView, View view, int i, long l) {
            //set the text of TextView
        }

        @Override
        public void onNothingSelected(AdapterView<?> adapterView) {

        }
    });

yourSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
        @Override
        public void onItemSelected(AdapterView<?> adapterView, View view, int i, long l) {
            yourSpinner.setOnItemSelectedListener(listener);
        }

        @Override
        public void onNothingSelected(AdapterView<?> adapterView) {

        }
    });

起初我创建了一个监听器并归因于一个变量回调;然后我创建了第二个匿名监听器,当第一次调用它时,这会改变监听器=]

于 2014-06-11T02:30:05.700 回答
4

创建一个布尔字段

private boolean inispinner;

在活动的 oncreate 内部

    spinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
        @Override
        public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
            if (!inispinner) {
                inispinner = true;
                return;
            }
            //do your work here
        }

        @Override
        public void onNothingSelected(AdapterView<?> parent) {

        }
    });
于 2019-02-28T11:47:09.533 回答
3

启用多个微调器的类似简单解决方案是将 AdapterView 放在一个集合中 - 在活动超类中 - 在第一次执行 onItemSelected(...) 时,然后在执行之前检查 AdapterView 是否在集合中。这启用了超类中的一组方法,并支持多个 AdapterView 和多个微调器。

超类...

private Collection<AdapterView> AdapterViewCollection = new ArrayList<AdapterView>();

   protected boolean firstTimeThrough(AdapterView parent) {
    boolean firstTimeThrough = ! AdapterViewCollection.contains(parent);
    if (firstTimeThrough) {
       AdapterViewCollection.add(parent);
     }
    return firstTimeThrough;
   }

子类...

public void onItemSelected(AdapterView<?> parent, View view, int pos, long id) {
      if (! firstTimeThrough(parent)) {
        String value = safeString(parent.getItemAtPosition(pos).toString());
        String extraMessage = EXTRA_MESSAGE;
        Intent sharedPreferencesDisplayIntent = new         Intent(SharedPreferencesSelectionActivity.this,SharedPreferencesDisplayActivity.class);
    sharedPreferencesDisplayIntent.putExtra(extraMessage,value);
    startActivity(sharedPreferencesDisplayIntent);
  }
  // don't execute the above code if its the first time through
  // do to onItemSelected being called during view initialization.

}

于 2013-07-18T16:26:00.313 回答
3

然后可以在 onTouch 方法中将用户交互标志设置为 true,并在onItemSelected()处理了选择更改后重置。我更喜欢这个解决方案,因为用户交互标志是专门为微调器处理的,而不是为活动中可能影响所需行为的其他视图处理的。

在代码中:

为微调器创建侦听器:

public class SpinnerInteractionListener implements AdapterView.OnItemSelectedListener, View.OnTouchListener {

    boolean userSelect = false;

    @Override
    public boolean onTouch(View v, MotionEvent event) {
        userSelect = true;
        return false;
    }

    @Override
    public void onItemSelected(AdapterView<?> parent, View view, int pos, long id) {
        if (userSelect) { 
            userSelect = false;
            // Your selection handling code here
        }
    }

}

将侦听器作为 anOnItemSelectedListener和 an添加到微调器OnTouchListener

SpinnerInteractionListener listener = new SpinnerInteractionListener();
mSpinnerView.setOnTouchListener(listener);
mSpinnerView.setOnItemSelectedListener(listener);
于 2017-03-07T03:39:59.227 回答
2

试试这个

spinner.postDelayed(new Runnable() {
        @Override
        public void run() {
            addListeners();
        }
    }, 1000);.o
于 2019-05-21T21:55:12.523 回答
2

代码

spinner.setOnTouchListener(new View.OnTouchListener() { 
@Override public boolean onTouch(View view, MotionEvent motionEvent) { isSpinnerTouch=true; return false; }});

holder.spinner_from.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
            @Override
            public void onItemSelected(AdapterView<?> adapterView, View view, int slot_position, long l) {
                if(isSpinnerTouch)
                {
                    Log.d("spinner_from", "spinner_from");
                    spinnerItemClickListener.onSpinnerItemClickListener(position, slot_position, AppConstant.FROM_SLOT_ONCLICK_CODE);
                }
                else {

                }
            }

            @Override
            public void onNothingSelected(AdapterView<?> adapterView) {

            }
        });
于 2019-10-17T06:37:08.787 回答
1

您可以先通过 setOnTouchListener 然后在 onTouch 中通过 setOnItemSelectedListener 来实现

@Override
public boolean onTouch(final View view, final MotionEvent event) {
 view.setOnItemSelectedListener(this)
 return false;
}
于 2015-11-19T08:06:11.520 回答
1

这对我有用:

    spinner.setSelection(0, false);
    new Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                spinner.setOnItemSelectedListener(listener);
            }, 500);
于 2019-09-27T08:03:43.910 回答
1

根据 Abhi 的回答,我做了这个简单的听众

class SpinnerListener constructor(private val onItemSelected: (position: Int) -> Unit) : AdapterView.OnItemSelectedListener {

    private var selectionCount = 0

    override fun onNothingSelected(parent: AdapterView<*>?) {
        //no op
    }

    override fun onItemSelected(parent: AdapterView<*>?, view: View?, position: Int, id: Long) {
        if (selectionCount++ > 1) {
           onItemSelected(position)
        }
    }
}
于 2019-10-06T11:08:10.557 回答
1

您可以像这样创建自定义OnItemSelectedListener。我已经采取val check=0并且在onItemSelected()方法中我确实检查了它的计数是否为0?如果 0 表示它在初始化期间被调用。所以干脆忽略它。

我还调用了单独的抽象方法,onUserItemSelected()我将这个方法称为check > 0. 这对我来说非常好。

abstract class MySpinnerItemSelectionListener : AdapterView.OnItemSelectedListener {

abstract fun onUserItemSelected(parent: AdapterView<*>?, view: View?, position: Int, id: Long)
private var check = 0
override fun onItemSelected(
    parent: AdapterView<*>?,
    view: View,
    position: Int,
    id: Long
) {
    if (++check > 1) {
        onUserItemSelected(parent, view, position, id)
    }
}

override fun onNothingSelected(parent: AdapterView<*>?) {}

}

然后你可以像这样设置监听器。

mySpinner.onItemSelectedListener =  object : MySpinnerItemSelectionListener() {
        override fun onUserItemSelected(
            parent: AdapterView<*>?,
            view: View?,
            position: Int,
            id: Long
        ) {
            //your user selection spinner code goes here
        }

    }
于 2021-09-28T19:47:16.897 回答
0

有同样的问题,这对我有用:

我有 2 个微调器,我在初始化期间以及与其他控件交互期间或从服务器获取数据后更新它们。

这是我的模板:

public class MyClass extends <Activity/Fragment/Whatever> implements Spinner.OnItemSelectedListener {

    private void removeSpinnersListeners() {
        spn1.setOnItemSelectedListener(null);
        spn2.setOnItemSelectedListener(null);
    }

    private void setSpinnersListeners() {
        new Handler().postDelayed(new Runnable() {
            @Override
            public void run() {
                spn1.setOnItemSelectedListener(MyClass.this);
                spn2.setOnItemSelectedListener(MyClass.this);
            }
        }, 1);
    }

    @Override
    public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
        // Your code here
    }

    @Override
    public void onNothingSelected(AdapterView<?> parent) {

    }
}

当类启动时使用setSpinnersListeners()而不是直接设置监听器。

Runnable 将阻止微调器在您设置它们的值后立即触发 onItemSelected。

如果您需要更新微调器(在服务器调用等之后),请在更新行之前使用removeSpinnersListeners() ,并在更新行之后使用 setSpinnersListeners( )。这将防止 onItemSelected 在更新后触发。

于 2015-10-22T15:04:33.890 回答
0

对我来说,Abhi 的解决方案在 Api 级别 27 之前都非常有效。

但似乎从 Api 级别 28 及更高级别开始,设置侦听器时不会调用 onItemSelected(),这意味着永远不会调用 onItemSelected()。

因此,我添加了一个简短的 if 语句来检查 Api 级别:

public void onItemSelected(AdapterView<?> parent, View arg1, int pos,long id) {

            if(Build.VERSION.SDK_INT >= 28){ //onItemSelected() doesn't seem to be called when listener is set on Api 28+
                check = 1;
            }

            if(++check > 1) {
                //Do your action here
            }
        }

我认为这很奇怪,我不确定其他人是否也有这个问题,但就我而言,它运作良好。

于 2020-03-05T17:24:59.033 回答
0

我在 Spinner 的顶部放置了一个 TextView,大小和背景与 Spinner 相同,这样我就可以在用户点击它之前更好地控制它的外观。有了 TextView,我还可以使用 TextView 来标记用户何时开始交互。

我的 Kotlin 代码如下所示:

private var mySpinnerHasBeenTapped = false

private fun initializeMySpinner() {

    my_hint_text_view.setOnClickListener {
        mySpinnerHasBeenTapped = true //turn flag to true
        my_spinner.performClick() //call spinner click
    }

    //Basic spinner setup stuff
    val myList = listOf("Leonardo", "Michelangelo", "Rafael", "Donatello")
    val dataAdapter: ArrayAdapter<String> = ArrayAdapter<String>(this, android.R.layout.simple_spinner_dropdown_item, myList)
    my_spinner.adapter = dataAdapter

    my_spinner.onItemSelectedListener = object : OnItemSelectedListener {

        override fun onItemSelected(parent: AdapterView<*>?, view: View, position: Int, id: Long) {

            if (mySpinnerHasBeenTapped) { //code below will only run after the user has clicked
                my_hint_text_view.visibility = View.GONE //once an item has been selected, hide the textView
                //Perform action here
            }
        }

        override fun onNothingSelected(parent: AdapterView<*>?) {
            //Do nothing
        }
    }
}

布局文件看起来像这样,重要的部分是 Spinner 和 TextView 共享相同的宽度、高度和边距:

        <FrameLayout
            android:layout_width="match_parent"
            android:layout_height="wrap_content">

            <Spinner
                android:id="@+id/my_spinner"
                android:layout_width="match_parent"
                android:layout_height="35dp"
                android:layout_marginStart="10dp"
                android:layout_marginEnd="10dp"
                android:background="@drawable/bg_for_spinners"

                android:paddingStart="8dp"
                android:paddingEnd="30dp"
                android:singleLine="true" />

            <TextView
                android:id="@+id/my_hint_text_view"
                android:layout_width="match_parent"
                android:layout_height="35dp"                
                android:layout_marginStart="10dp"
                android:layout_marginEnd="10dp"
                android:background="@drawable/bg_for_spinners"

                android:paddingStart="8dp"
                android:paddingEnd="30dp"
                android:singleLine="true"
                android:gravity="center_vertical"
                android:text="*Select A Turtle"
                android:textColor="@color/green_ooze"
                android:textSize="16sp" />

        </FrameLayout>

我确信其他解决方案可以在您忽略第一个 onItemSelected 调用的情况下工作,但我真的不喜欢假设它总是会被调用的想法。

于 2020-06-30T16:13:47.667 回答