With Android we haven't a classic ComboBox like in other frameworks but we have instead a Spinner.
Actually it's exactly the same and only the name differs.
In this Android Spinner tutorial we're going to create our own cutom Spinner by replacing classic texts with squares of color.
The minimum SDK for this tutorial is the API 14 (Android 4.0 or also known as IceCreamSandwich).
Spinner, Object class and 2's complement will be see in this example.
The MainActiviy is quite simple and so easy to understand.
We have then the BaprogArrayAdapter which is a class that extends the Android ArrayAdapter.
Within we're going to fill an array of Integers (list of colors) and use this array to display every element (color) in this array.
We use Integer and not int type because we can't use primitive types, indeed Integer inherits from Object whereas int doesn't.
In our example we don't specify the type when we declare our ArrayAdapter but as this class is a template one we could have replaced Object[] objects with Integer[] objects in the constructor parameters.
And thus not having to cast it.
The FirstFragment contains data needed to create our Spinner of colors.
We start by creating an array of Integers in which we add the 3 colors present in the file colors.xml.
Then we link our BadprogArrayAdapter with this Fragment.
And finally when the user will click on a color, this color will be automatically changed in the View in the center of the Fragment.
It's interesting to notice here that the value gets from the Spinner.getSelectedItem() method is an Object type.
It's an ID that only the Android OS knows how to use it.
So we can't directly use this Object like that, we have to cast it.
In our case we cast it into an int.
Once converted into an int, we can't either use it, we have to call the ContextCompat.getColor() method in order to retrieve a decimal value.
This decimal value could be for example "-16777216".
Impossible to understand what it is except if we convert it in HEX format with two's complement.
That is #FF000000 (black color).
// badprog.com package com.badprog.badprogspinnercolors; import android.os.Bundle; import androidx.appcompat.app.AppCompatActivity; import androidx.appcompat.widget.Toolbar; // =============================================================================================== // // =============================================================================================== public class MainActivity extends AppCompatActivity { // =============================================================================================== // // =============================================================================================== @Override protected void onCreate(Bundle savedInstanceState) { // super.onCreate(savedInstanceState); // setContentView(R.layout.activity_main); // Toolbar toolbar = findViewById(R.id.toolbar); setSupportActionBar(toolbar); } }
// badprog.com package com.badprog.badprogspinnercolors; import android.content.Context; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.ArrayAdapter; import androidx.annotation.NonNull; // =============================================================================================== // // =============================================================================================== public class BadprogArrayAdapter extends ArrayAdapter { private Integer[] _listOfColors; // =============================================================================================== // // =============================================================================================== public BadprogArrayAdapter(@NonNull Context context, int resource, @NonNull Object[] objects) { // super(context, resource, objects); // _listOfColors = (Integer[]) objects; } // =============================================================================================== // // =============================================================================================== @Override public View getDropDownView(int position, View convertView, ViewGroup parent) { // return getBadprogView(position, convertView, parent); } // =============================================================================================== // // =============================================================================================== @Override public View getView(int position, View convertView, ViewGroup parent) { // return getBadprogView(position, convertView, parent); } // =============================================================================================== // // =============================================================================================== public View getBadprogView(int position, View convertView, ViewGroup parent) { // LayoutInflater inflater = LayoutInflater.from(getContext()); View viewXml = inflater.inflate(R.layout.badprog_spinner_color, parent, false); // View viewElement = viewXml.findViewById(R.id.view); viewElement.setBackgroundColor(getContext().getResources().getColor(_listOfColors[position])); // return viewXml; } }
// badprog.com package com.badprog.badprogspinnercolors; import android.os.Bundle; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.Spinner; import androidx.annotation.NonNull; import androidx.core.content.ContextCompat; import androidx.fragment.app.Fragment; // =============================================================================================== // // =============================================================================================== public class FirstFragment extends Fragment implements AdapterView.OnItemSelectedListener { // private View _viewDisplay; private Spinner _spinner; Integer _arrayOfColors [] = { R.color.colorAccent, R.color.colorPrimary, R.color.colorPrimaryDark }; // =============================================================================================== // // =============================================================================================== @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { // Inflate the layout for this fragment return inflater.inflate(R.layout.fragment_first, container, false); } // =============================================================================================== // // =============================================================================================== public void onViewCreated(@NonNull View view, Bundle savedInstanceState) { // super.onViewCreated(view, savedInstanceState); // ArrayAdapter arrayAdapter = new BadprogArrayAdapter(FirstFragment.this.getContext(), android.R.layout.simple_spinner_item, _arrayOfColors); arrayAdapter.setDropDownViewResource(android.R.layout.select_dialog_multichoice); // _spinner = view.findViewById(R.id.spinner_colors); _spinner.setOnItemSelectedListener(this); _spinner.setAdapter(arrayAdapter); // _viewDisplay = view.findViewById(R.id.view_display); } // =============================================================================================== // // =============================================================================================== @Override public void onItemSelected(AdapterView<?> parent, View view, int position, long id) { // Object object = _spinner.getSelectedItem(); int color = ContextCompat.getColor(getContext(), (int)object); // _viewDisplay.setBackgroundColor(color); } // =============================================================================================== // // =============================================================================================== @Override public void onNothingSelected(AdapterView<?> parent) { // } }
<?xml version="1.0" encoding="utf-8"?> <androidx.coordinatorlayout.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity"> <com.google.android.material.appbar.AppBarLayout android:layout_width="match_parent" android:layout_height="wrap_content" android:theme="@style/AppTheme.AppBarOverlay"> <androidx.appcompat.widget.Toolbar android:id="@+id/toolbar" android:layout_width="match_parent" android:layout_height="?attr/actionBarSize" android:background="?attr/colorPrimary" app:popupTheme="@style/AppTheme.PopupOverlay" /> </com.google.android.material.appbar.AppBarLayout> <include layout="@layout/content_main" /> </androidx.coordinatorlayout.widget.CoordinatorLayout>
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="wrap_content" android:orientation="horizontal" android:padding="5sp" > <View android:id="@+id/view" android:layout_width="match_parent" android:layout_height="50dp" android:text="Button" /> </LinearLayout>
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_width="match_parent" android:layout_height="match_parent" app:layout_behavior="@string/appbar_scrolling_view_behavior"> <fragment android:id="@+id/nav_host_fragment" android:name="androidx.navigation.fragment.NavHostFragment" android:layout_width="0dp" android:layout_height="0dp" app:defaultNavHost="true" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" app:navGraph="@navigation/nav_graph" /> </androidx.constraintlayout.widget.ConstraintLayout>
<?xml version="1.0" encoding="utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".FirstFragment"> <Spinner android:id="@+id/spinner_colors" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_marginStart="16dp" android:layout_marginLeft="16dp" android:layout_marginTop="16dp" android:layout_marginEnd="16dp" android:layout_marginRight="16dp" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" /> <View android:id="@+id/view_display" android:layout_width="100dp" android:layout_height="100dp" android:text="Button" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toBottomOf="@+id/spinner_colors" /> </androidx.constraintlayout.widget.ConstraintLayout>
<menu xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" tools:context="com.badprog.badprogspinnercolors.MainActivity"> <item android:id="@+id/action_settings" android:orderInCategory="100" android:title="Title" app:showAsAction="never" /> </menu>
<?xml version="1.0" encoding="utf-8"?> <navigation xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/nav_graph" app:startDestination="@id/FirstFragment"> <fragment android:id="@+id/FirstFragment" android:name="com.badprog.badprogspinnercolors.FirstFragment" tools:layout="@layout/fragment_first"> </fragment> </navigation>
<?xml version="1.0" encoding="utf-8"?> <resources> <color name="colorPrimary">#6200EE</color> <color name="colorPrimaryDark">#3700B3</color> <color name="colorAccent">#03DAC5</color> </resources>
<resources> <dimen name="fab_margin">16dp</dimen> </resources>
<resources> <string name="app_name">BadprogSpinnerColors</string> </resources>
<resources> <!-- Base application theme. --> <style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar"> <!-- Customize your theme here. --> <item name="colorPrimary">@color/colorPrimary</item> <item name="colorPrimaryDark">@color/colorPrimaryDark</item> <item name="colorAccent">@color/colorAccent</item> </style> <style name="AppTheme.NoActionBar"> <item name="windowActionBar">false</item> <item name="windowNoTitle">true</item> </style> <style name="AppTheme.AppBarOverlay" parent="ThemeOverlay.AppCompat.Dark.ActionBar" /> <style name="AppTheme.PopupOverlay" parent="ThemeOverlay.AppCompat.Light" /> </resources>
You are now able to create your own Spinner in which you can add everything you want.
For example adding icons instead of colors should be a good idea.
Anyway, good job you did it.
Add new comment