Android - Gridview con Custom BaseAdapter, crea onclicklistener [duplicato]
-
26-12-2019 - |
Domanda
Ho creato una visualizzazione a griglia che visualizza le lettere dell'alfabeto.Popolo la visualizzazione della griglia con un array di stringhe utilizzando un BaseAdapter personalizzato.
Quello che voglio fare è poter ottenere il valore (lettera) della cella cliccata.Per verificare che funzioni, ho creato un TextView e voglio che quando l'utente fa clic su un elemento (cella) imposti il testo del TextView con il valore della cella selezionata
Ho fatto un tentativo che non funziona.Ecco il mio codice:
attività_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/container"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity"
tools:ignore="MergeRootFrame" >
<GridView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/myGridView"
android:layout_alignParentTop="true"
android:layout_alignParentLeft="true"
android:numColumns="7" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:textAppearance="?android:attr/textAppearanceLarge"
android:text="Large Text"
android:id="@+id/feedback"
android:layout_alignParentBottom="true"
android:layout_centerHorizontal="true" />
</RelativeLayout>
cella.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="wrap_content"
android:layout_height="wrap_content">
<Button
android:id="@+id/grid_item"
android:layout_gravity="center"
android:textAppearance="?android:attr/textAppearanceMedium"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="05"
android:textSize="20sp"
android:enabled="true"
android:clickable="true">
</Button>
</LinearLayout>
Attività principale
public class MainActivity extends Activity {
private TextView text;
private GridView gridView;
private final String[] items = new String[]{"A", "B", "C", "D", "E", "F", "G", "H", "I", "J",
"K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z"};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
text = (TextView) findViewById(R.id.feedback);
gridView = (GridView) this.findViewById(R.id.myGridView);
CustomGridAdapter gridAdapter = new CustomGridAdapter(MainActivity.this, items);
gridView.setAdapter(gridAdapter);
gridView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
text.setText((String) (gridView.getItemAtPosition(position)));
Log.i("ITEM_CLICKED", "" + (String) (gridView.getItemAtPosition(position)));
}
});
}
}
CustomGridAdapter
public class CustomGridAdapter extends BaseAdapter {
private Context context;
private String[] items;
LayoutInflater inflater;
public CustomGridAdapter(Context context, String[] items) {
this.context = context;
this.items = items;
inflater = (LayoutInflater) this.context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = inflater.inflate(R.layout.cell, null);
}
Button button = (Button) convertView.findViewById(R.id.grid_item);
button.setText(items[position]);
return convertView;
}
@Override
public int getCount() {
return items.length;
}
@Override
public Object getItem(int position) {
return items[position];
}
@Override
public long getItemId(int position) {
return position;
}
}
Che cosa sto facendo di sbagliato?
Modificare:Sembra che gridView.setOnItemClickListener(...);non viene chiamato/eseguito.L'ho cambiato in questo per controllarlo e non vedo nulla nel logcat.
gridView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Log.i("ON_ITEM_CLICK_LISTENER", "item clicked");
}
});
Soluzione
Il problema nel tuo caso è che il pulsante consuma il clic e non arriva a GridView.Vedo due opzioni:
- Quello facile:In
cell.xml
avere queste proprietà per il tuoButton
:android:clickable="false"
Eandroid:focusable="false"
- A grosso modo:invia l'attività come parametro all'adattatore ed esponi un metodo pubblico da
MainActivity
.Se pensi di riutilizzare l'adattatore, invia un tipo di attività astratta o un'interfaccia.
Qualcosa come:
public class MainActivity extends Activity {
private TextView text;
private GridView gridView;
private final String[] items = new String[]{"A", "B", "C", "D", "E", "F", "G", "H", "I", "J",
"K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z"};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
text = (TextView) findViewById(R.id.feedback);
gridView = (GridView) this.findViewById(R.id.myGridView);
CustomGridAdapter gridAdapter = new CustomGridAdapter(MainActivity.this, items);
gridView.setAdapter(gridAdapter);
}
public void itemClicked(int position) {
text.setText(items[position]);
}
}
e nel tuo adattatore:
public class CustomGridAdapter extends BaseAdapter {
private MainActivity context;
private String[] items;
LayoutInflater inflater;
public CustomGridAdapter(MainActivity context, String[] items) {
this.context = context;
this.items = items;
inflater = (LayoutInflater) this.context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
}
public View getView(final int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = inflater.inflate(R.layout.cell, null);
}
Button button = (Button) convertView.findViewById(R.id.grid_item);
button.setText(items[position]);
button.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
context.itemClicked(position);
}
});
return convertView;
}
@Override
public int getCount() {
return items.length;
}
@Override
public Object getItem(int position) {
return items[position];
}
@Override
public long getItemId(int position) {
return position;
}
}
Altri suggerimenti
In realtà per gridview setOnItemClickListener non stai facendo nulla di sbagliato.È la visualizzazione della cella che crea il problema.Prova a cambiare il pulsante di visualizzazione cella in una visualizzazione testo, dovrebbe funzionare.In questo momento, quando tocchi un elemento in una visualizzazione a griglia, l'elemento non viene effettivamente cliccato perché il pulsante di visualizzazione cella ha la propria implementazione di OnClickListener, ed è per questo che setOnItemClickListener non viene chiamato.
Puoi utilizzare setTag() e getTag() che ti consentono di inserire/ottenere qualsiasi tipo di oggetto.Quindi in futuro sarai in grado di passare qualsiasi cosa (se dopo farai il cast appropriato)
public View getView(int position, View convertView, ViewGroup parent) {
if (convertView == null) {
convertView = inflater.inflate(R.layout.cell, null);
}
Button button = (Button) convertView.findViewById(R.id.grid_item);
button.setText(items[position]);
convertView.setTag(items[position]);
return convertView;
}
E
gridView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
String str = (String) view.getTag();
text.setText(str);
Log.i("ITEM_CLICKED", "" + (String) (gridView.getItemAtPosition(position)));
}
});