NullPointerException для view.findViewById() с анализатором запросов анализа /ListView
Я заполнил ListView, используя пользовательский адаптер с Parse Query Adapter. Моя проблема в том, что я могу только получить доступ к первому элементу списка. В ListView у меня есть кнопка, которую я хочу вызвать функцию.
android:id="@+id/showGrunts"
android:layout_height="wrap_content"
android:layout_width="wrap_content"
android:layout_below="@id/buttonthang"
android:clickable="true"
android:onClick="displayGruntList"
А вот функция, которая вызывается при нажатии кнопки:
public void displayGruntList(View view) {
TextView objectID = (TextView) findViewById(R.id.tvPostIDCheese);
String id = objectID.getText().toString();
//String id = ((TextView)(view.findViewById(R.id.tvPostIDCheese))).getText().toString();
Intent i = new Intent(GruntrMain.this, GruntList.class);
Bundle idData = new Bundle();
idData.putString("id", id);
i.putExtras(idData);
startActivity(i);
}
Обратите внимание на закомментированную строку. Моя проблема в том, что если я нажму эту кнопку, она будет отображать только связанные элементы для первого элемента ListView, даже если я нажму на кнопку из другого элемента в списке. Я был в состоянии определить это, отображая ID как тост, как только он получен. Независимо от того, что я нажимаю, он всегда показывает идентификатор верхнего элемента в списке. Это остается верным, если я переупорядочиваю их, и, отображая идентификаторы в каждом элементе списка, я уверен, что это не проблема в моем запросе, поскольку все они отображаются с их правильными идентификаторами.
Мое понимание представления, которое передается в функцию, заключается в том, что оно исходит из фактического элемента, который выбирается. Я видел другой код, который использует view.findViewById(), но когда я пытаюсь это сделать, строка дает мне исключение NullPointerException. То, что я хочу сделать, это получить текст (который представляет собой идентификатор) из TextView для этого конкретного элемента. Опять же, с кодом, который у меня есть на данный момент, он извлекает только текст из первого элемента. Так что проблема не в моем Parse Query, а в том, как я к нему обращаюсь. Весь файл.java, в котором возникает эта проблема, можно увидеть ниже.
package com.snowflakes.gruntr;
import java.util.List;
import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.ImageView;
import android.widget.ListView;
import android.widget.TextView;
import com.parse.FindCallback;
import com.parse.GetCallback;
import com.parse.ParseException;
import com.parse.ParseObject;
import com.parse.ParseQuery;
import com.parse.ParseQueryAdapter;
import com.parse.ParseRelation;
import com.parse.ParseUser;
public class GruntrMain extends Activity {
private static final int LOGIN_REQUEST = 0;
private ListView listView;
private ParseQueryAdapter<ParseObject> mainAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
// TODO Auto-generated method stub
super.onCreate(savedInstanceState);
setContentView(R.layout.gruntr_main);
ImageView addF = (ImageView) findViewById(R.id.ivAddFriend);
final ImageView fReq = (ImageView) findViewById(R.id.ivFriendRequest);
ImageView mPost = (ImageView) findViewById(R.id.ivMakePost);
mainAdapter = new CustomAdapter(this);
listView = (ListView) findViewById(R.id.lvPostList);
listView.setAdapter(mainAdapter);
mainAdapter.loadObjects();
Thread refreshFeed = new Thread() {
public void run() {
while (true) {
try {
sleep(20000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
mainAdapter.loadObjects();
}
}
}// public void run()
}; // new Thread()
refreshFeed.start();
// This thread is used for friend polling
// In order to get it to run repeatedly, may need to make an infinite
// loop
// If a friend request is found to be pending, it will cause a button to
// appear, allowing you to accept the friend request
// For later implementation, this can be refined
// For example: It can cause the notification area to show that you have
// a friend request pending
// Also, the ability to deny friend requests is already built in. I'll
// actually code that in when it become relevant
// -- Aaron
Thread friendRequestPolling = new Thread() {
public void run() {
while (true) {
try {
sleep(4000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
ParseUser user = ParseUser.getCurrentUser();
ParseQuery<ParseObject> friendQuery = ParseQuery
.getQuery("FriendRequests");
friendQuery.whereEqualTo("Recipient", user);
friendQuery
.getFirstInBackground(new GetCallback<ParseObject>() {
public void done(ParseObject request,
ParseException e) {
if (request != null) {
fReq.setImageResource(R.drawable.friendrequesttrue);
} else {
fReq.setImageResource(R.drawable.friendrequest);
}
}
});
}
}
}// public void run()
}; // new Thread()
friendRequestPolling.start();
Thread friendAcceptedPolling = new Thread() {
public void run() {
while (true) {
try {
sleep(4000);
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
ParseUser user = ParseUser.getCurrentUser();
ParseQuery<ParseObject> friendQuery = ParseQuery
.getQuery("FriendRequests");
friendQuery.whereEqualTo("Sender", user);
friendQuery.whereEqualTo("Accepted", "true");
friendQuery
.findInBackground(new FindCallback<ParseObject>() {
public void done(
List<ParseObject> returnedUsers,
ParseException e) {
for (ParseObject currentUser : returnedUsers) {
if (currentUser != null) {
ParseUser user = ParseUser
.getCurrentUser();
ParseRelation<ParseUser> relation = user
.getRelation("friends");
relation.add(currentUser
.getParseUser("Recipient"));
user.saveInBackground();
currentUser
.deleteInBackground();
}
}
}
});
}
}
}// public void run()
}; // new Thread()
friendAcceptedPolling.start();
mPost.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Intent i = new Intent(GruntrMain.this, MakePost.class);
startActivity(i);
}
});
addF.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Intent i = new Intent(GruntrMain.this, AddFriend.class);
startActivity(i);
}
});
fReq.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
Intent i = new Intent(GruntrMain.this, FriendRequests.class);
startActivity(i);
}
});
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == LOGIN_REQUEST) {
if (resultCode == RESULT_OK) {
Intent i = new Intent(GruntrMain.this, GruntrMain.class);
i.putExtra("User", ParseUser.getCurrentUser().getString("name"));
startActivity(i);
finish();
} else {
Intent i = new Intent(GruntrMain.this, Goodbye.class);
startActivity(i);
finish();
}
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
// Inflate the menu; this adds items to the action bar if it is present.
getMenuInflater().inflate(R.menu.main_menu, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
// Handle action bar item clicks here. The action bar will
// automatically handle clicks on the Home/Up button, so long
// as you specify a parent activity in AndroidManifest.xml.
int id = item.getItemId();
if (id == R.id.action_settings) {
return true;
} else if (id == R.id.action_logout) {
ParseUser.logOut();
ParseUser currentUser = ParseUser.getCurrentUser();
ParseLoginBuilder builder = new ParseLoginBuilder(GruntrMain.this);
Intent parseLoginIntent = builder
.setAppLogo(R.drawable.orangebanner)
.setParseLoginEnabled(true)
.setParseLoginButtonText("Go")
.setParseSignupButtonText("Register")
.setParseLoginHelpText("Forgot password?")
.setParseLoginInvalidCredentialsToastText(
"Your email and/or password is not correct")
.setParseLoginEmailAsUsername(true)
.setParseSignupSubmitButtonText("Submit registration")
.build();
startActivityForResult(parseLoginIntent, 0);
} else if (id == R.id.action_post) {
Intent i = new Intent(GruntrMain.this, MakePost.class);
startActivity(i);
}
return super.onOptionsItemSelected(item);
}
public void displayGruntList(View view) {
TextView objectID = (TextView) findViewById(R.id.tvPostIDCheese);
String id = objectID.getText().toString();
// String id =
// ((TextView)(view.findViewById(R.id.tvPostIDCheese))).getText().toString();
Intent i = new Intent(GruntrMain.this, GruntList.class);
Bundle idData = new Bundle();
idData.putString("id", id);
i.putExtras(idData);
startActivity(i);
}
public void gruntAtIt(View view) {
TextView objectID = (TextView) findViewById(R.id.tvPostIDCheese);
String id = objectID.getText().toString();
ParseQuery<ParseObject> query = new ParseQuery<ParseObject>("Post");
query.whereEqualTo("objectId", id);
ParseObject post = null;
try {
post = query.getFirst();
} catch (ParseException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
if (post != null) {
post.increment("Grunts");
post.saveInBackground();
}
}
Intent i = new Intent(GruntrMain.this, RecordGrunt.class);
Bundle idData = new Bundle();
idData.putString("id", id);
i.putExtras(idData);
startActivity(i);
}
}
Вот CustomAdapter.java, где выполняется запрос и заполняется список.
package com.snowflakes.gruntr;
import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;
import com.parse.ParseFile;
import com.parse.ParseImageView;
import com.parse.ParseObject;
import com.parse.ParseQuery;
import com.parse.ParseQueryAdapter;
import com.parse.ParseRelation;
import com.parse.ParseUser;
public class CustomAdapter extends ParseQueryAdapter<ParseObject> {
public CustomAdapter(Context context) {
// Use the QueryFactory to construct a PQA that will only show
// Todos marked as high-pri
super(context, new ParseQueryAdapter.QueryFactory<ParseObject>() {
public ParseQuery create() {
ParseUser user = ParseUser.getCurrentUser();
ParseRelation<ParseUser> relation = user.getRelation("friends");
ParseQuery friendsQuery = relation.getQuery();
ParseQuery<ParseObject> query = new ParseQuery<ParseObject>("Post");
query.whereMatchesQuery("CreatedBy", friendsQuery);
query.addDescendingOrder("createdAt");
return query;
}
});
}
// Customize the layout by overriding getItemView
@Override
public View getItemView(ParseObject object, View v, ViewGroup parent) {
if (v == null) {
v = View.inflate(getContext(), R.layout.individual_post, null);
}
super.getItemView(object, v, parent);
// Add and download the image
ParseImageView todoImage = (ParseImageView) v.findViewById(R.id.icon);
ParseFile imageFile = object.getParseFile("Image");
if (imageFile != null) {
todoImage.setParseFile(imageFile);
todoImage.loadInBackground();
}
// Add the title view
TextView titleTextView = (TextView) v.findViewById(R.id.text1);
titleTextView.setText(object.get("Name").toString());
// Add a reminder of how long this item has been outstanding
TextView timestampView = (TextView) v.findViewById(R.id.timestamp);
timestampView.setText(object.getCreatedAt().toString());
TextView objectID = (TextView) v.findViewById(R.id.tvPostIDCheese);
objectID.setText(object.getObjectId());
TextView gruntCount = (TextView) v.findViewById(R.id.showGrunts);
gruntCount.setText("Show Grunts (" + object.getNumber("Grunts") + ")");
return v;
}
}
1 ответ
Аргумент view, передаваемый вашему onClick, - это просто представление, на которое был выполнен щелчок, а не окружающий макет. Если tvPostIDCheese не является дочерним видом вашей кнопки, он не доступен таким образом.
Вместо этого попробуйте сохранить свой идентификатор объекта в представлении вашего кликабельного с помощью gruntCount.setTag(object.getObjectId())
во время getItemView вашего адаптера. Затем в displayGruntList получить его с (String)view.getTag()
,