Как я могу заполнить ListView на SecondActivity данными из FirstActivity в Android? - PullRequest
0 голосов
/ 27 мая 2020

Я работаю над проектом приложения для поиска книг. Пользователь вводит название, и приложение ищет его в Google Книгах. Изначально у меня было одно занятие и макет. Я решил использовать два макета (один для ввода заголовка пользователем, а другой отображает результаты); пришлось создать другое действие, потому что макет результатов генерировал исключение, касающееся 'нулевой' ссылки на объект . После создания второго действия я использовал намерение для передачи данных списка между двумя действиями; как только я узнал, что для этой цели мне нужно использовать объект Serialize или Parcelable . Следуя инструкциям в Передать список объектов из одного действия в другое в android и https://www.vogella.com/tutorials/AndroidParcelable/article.html, мне не удалось реализовать оба, потому что Serialize отправил пустой ArrayList ( таким образом, пустая страница результатов, даже если есть попадания в книги) и Parcelable каждый раз выбрасывал разные исключения, когда я его использовал , вероятно, потому что я использую ArrayAdapter для заполнения ListView книгами . Я не думаю, что готов реализовать API на https://www.vogella.com/tutorials/AutoValue/article.html и также не могу использовать фрагменты. Лучшей структурой было бы использовать одно действие, View.GONE фоновое изображение, которое я так люблю sh, и отображать список объектов книги под полем поиска. Но было бы неплохо сделать продолжение Serialize & Parcelable - для будущих проектов. Итак, товарищи-разработчики, как решить эту проблему?

Снимок экрана FirstActivity Снимок экрана SecondActivity

Вот мой код: MainActivity:

package project.android.gbookslisting;

import android.app.LoaderManager;
import android.content.Intent;
import android.content.Loader;
import android.graphics.Color;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;

import java.io.Serializable;
import java.util.ArrayList;
import java.util.List;

import androidx.appcompat.app.AppCompatActivity;

import static project.android.gbookslisting.ResultsActivity.adapter;

//import android.support.v4.content.AsyncTaskLoader;
//import android.support.v7.app.AppCompatActivity;

public class ParamsActivity extends AppCompatActivity implements LoaderManager.LoaderCallbacks<List<Book>>, Serializable {

    private static final int LOADER_ID = 0;
    private static String LOG_TAG = ParamsActivity.class.getName();
    EditText text;
    String query;
    private LoaderManager loaderManager = getLoaderManager();

    @Override
    protected void onCreate (Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.parameters);


        Button query = findViewById(R.id.search_button);
        text = findViewById(R.id.deets_field);


        query.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick (View view) {
                if (text.getText().toString().length() > 0) {                    loaderManager.initLoader(LOADER_ID, null, ParamsActivity.this);
                } else if (text.getText().length() < 1) {
                    text.setHint("Please enter book title/details");
                    text.setHintTextColor(Color.RED);
                }
            }
        });

    }

    @Override
    public Loader<List<Book>> onCreateLoader (int i, Bundle bundle) {
        query = text.getText().toString();
        return new BookLoader(this, query);
    }

    @Override
    public void onLoadFinished (Loader<List<Book>> loader, List<Book> data) {
        // If there is a valid list of {@link Book}s, then add them to the adapter's dataset. This will trigger the ListView to update.
        if (data != null && !data.isEmpty()) {

            data = new ArrayList<Book>();

            Intent i = new Intent(getApplicationContext(), ResultsActivity.class);
            i.putExtra("data", (Serializable) data);
            startActivity(i);
        }
    }

    @Override
    public void onLoaderReset (Loader loader) {
        adapter = new Adapt(this, new ArrayList<Book>());
        adapter.clear();
    }

}

SecondActivity:

package project.android.gbookslisting;

import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AdapterView;
import android.widget.ListView;
import android.widget.TextView;

import java.io.Serializable;
import java.util.ArrayList;

import androidx.appcompat.app.AppCompatActivity;

public class ResultsActivity extends AppCompatActivity implements Serializable {

    static Adapt adapter;
    static TextView emptyResult;
    ListView bookEntries;
    String LOG_TAG = ResultsActivity.class.getName();

    @Override
    protected void onCreate (Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.hits_page);

        Intent i = getIntent();
        ArrayList<Book> books = (ArrayList<Book>) i.getSerializableExtra("data");

        emptyResult = findViewById(R.id.matches_nill);
        emptyResult.setText(R.string.matches0);

        if (!books.isEmpty()) {
            emptyResult.setVisibility(View.GONE);

            // Create a new adapter that takes a rich (or otherwise empty) list of books as input
            adapter = new Adapt(this, new ArrayList<Book>());

            // Get the list of books from {@link Search}
            bookEntries = findViewById(R.id.catalog);           

            bookEntries.setAdapter(adapter);
            bookEntries.setEmptyView(emptyResult);

            bookEntries.setOnItemClickListener(new AdapterView.OnItemClickListener() {
                @Override
                public void onItemClick (AdapterView<?> adapterView, View view, int position, long l) {
                    // Find the current book that was clicked on
                    Book currentBook = adapter.getItem(position);

                    // Convert the String URL into a URI object (to pass into the Intent constructor)
                    Uri bookUri = Uri.parse(currentBook.getPage());

                    // Create a new intent to view the book URI
                    Intent websiteIntent = new Intent(Intent.ACTION_VIEW, bookUri);

                    // Send the intent to launch a new activity
                    startActivity(websiteIntent);
                }
            });

//            adapter.clear();

            adapter.addAll(books);
        } else {
            emptyResult.setVisibility(View.VISIBLE);
            emptyResult.setText(R.string.matches0);
        }

    }
}

Книга объект:

package project.android.gbookslisting;

import java.io.Serializable;
import java.util.Date;

public class Book {

    private String book_title;
    private String author;
    private String publishing_year;
    private String page;


    public Book (String theTitle, String theAuthor, String theYear, String thePage) {
        this.book_title = theTitle;
        this.author = theAuthor;
        this.publishing_year = theYear;
        this.page = thePage;
    }


    public Book setBook_title (String book_title) {
        this.book_title = book_title;
        return this;
    }

    public Book setAuthor (String author) {
        this.author = author;
        return this;
    }

    public Book setPublishing_year (String publishing_year) {
        this.publishing_year = publishing_year;
        return this;
    }

    public Book setPage (String page) {
        this.page = page;
        return this;
    }

    protected String getAuthor () {
        return author;
    }

    protected String getPublishing_year () { return publishing_year; }

    protected String getPage () {
        return page;
    }

    protected String getBook_title () {
        return book_title;
    }
}

Ответы [ 2 ]

0 голосов
/ 29 мая 2020

Хорошо, поэтому я смог исправить свою реализацию интерфейса Parcelable , и теперь FirstActivity передает данные в SecondActivity . Вот он:

package project.android.gbookslisting;

import android.os.Parcel;
import android.os.Parcelable;

public class Book implements Parcelable {

    private String book_title;
    private String author;
    private String publishing_year;
    private String page;


    public Book (String theTitle, String theAuthor, String theYear, String thePage) {
        this.book_title = theTitle;
        this.author = theAuthor;
        this.publishing_year = theYear;
        this.page = thePage;
    }

    public Book(Parcel in) {
        this.book_title = in.readString();
        this.author = in.readString();
        this.publishing_year = in.readString();
        this.page = in.readString();
    }

    public static final Creator<Book> CREATOR = new Creator<Book>() {
        @Override
        public Book createFromParcel (Parcel in) {
            return new Book(in);
        }

        @Override
        public Book[] newArray (int size) {
            return new Book[size];
        }
    };

    protected String getAuthor () { return author; }

    protected String getPublishing_year () { return publishing_year; }

    protected String getPage () {
        return page;
    }

    protected String getBook_title () {
        return book_title;
    }

    @Override
    public int describeContents () {
        return 0;
    }

    @Override
    public void writeToParcel (Parcel parcel, int flags) {
        parcel.writeString(book_title);
        parcel.writeString(author);
        parcel.writeString(publishing_year);
        parcel.writeString(page);
    }

}

Я опубликовал репозиторий на GitHub @ GBooks Listing , ожидая дополнительных функций / обновлений.

ResultsActivity Конечно, выглядит неубедительно, но я улучшу тематику и все, что влияет на внешний вид. Спасибо.

0 голосов
/ 27 мая 2020

Вы можете объявить переменную, которая будет обрабатывать ваш data = new ArrayList<Book>(); как public static .. Однако это не рекомендуется для высокоуровневой кодовой базы, но для ваших структур вы можете реализовать ее.

просто объявите переменную, подобную этой выше protected void onCreate() в FirstActivity ..

public static List<Book> book_data = new ArrayList<>();

и перенесите data из public void onLoadFinished (Loader<List<Book>> loader, List<Book> data) в book_data

@Override
public void onLoadFinished (Loader<List<Book>> loader, List<Book> data) {
    // If there is a valid list of {@link Book}s, then add them to the adapter's dataset. This will trigger the ListView to update.
    if (data != null && !data.isEmpty()) {

        //data = new ArrayList<Book>();
        book_data = data;

        Intent i = new Intent(getApplicationContext(), ResultsActivity.class);
        startActivity(i);
    }
}`

Вы также можете попробовать отладить его, если он содержит значение Log.d(TAG, String.valueOf(data.size()));

и в SecondActivity

@Override
protected void onCreate (Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.hits_page);

    List<Book> books = new ArrayList<>();
    books = FirstActivity.book_data; //Get the data from the First Activity

    ....
    ....
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...