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(),

Смотрите setTag и getTag для дальнейшего использования.