web-dev-qa-db-de.com

Rufen Sie den Kontext einer Aktivität aus der Ansichtsmodellklasse ab

Mein Code basiert auf einem Beispiel, das Android-Architekturkomponenten und Datenbindung verwendet. Dies ist ein neuer Weg für mich, und die Art und Weise, wie er codiert wird, macht es schwierig, eine neue Aktivität mit den Informationen des angeklickten Posts richtig zu öffnen.

Dies ist der Adapter der Pfosten

class PostListAdapter : RecyclerView.Adapter<PostListAdapter.ViewHolder>() {
    private lateinit var posts: List<Post>

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): PostListAdapter.ViewHolder {
        val binding: ItemPostBinding = DataBindingUtil.inflate(
            LayoutInflater.from(parent.context),
            R.layout.item_post,
            parent, false
        )

        return ViewHolder(binding)
    }

    override fun onBindViewHolder(holder: PostListAdapter.ViewHolder, position: Int) {
        holder.bind(posts[position])
    }

    override fun getItemCount(): Int {
        return if (::posts.isInitialized) posts.size else 0
    }

    fun updatePostList(posts: List<Post>) {
        this.posts = posts
        notifyDataSetChanged()
    }

    inner class ViewHolder(private val binding: ItemPostBinding) : RecyclerView.ViewHolder(binding.root) {
        private val viewModel = PostViewModel()

        fun bind(post: Post) {
            viewModel.bind(post)
            binding.viewModel = viewModel
        }
    }
}

Die bind-Methode stammt aus der Ansichtsmodellklasse:

class PostViewModel : BaseViewModel() {
    private val image = MutableLiveData<String>()
    private val title = MutableLiveData<String>()
    private val body = MutableLiveData<String>()

    fun bind(post: Post) {
        image.value = post.image
        title.value = post.title
        body.value = post.body
    }

    fun getImage(): MutableLiveData<String> {
        return image
    }

    fun getTitle(): MutableLiveData<String> {
        return title
    }

    fun getBody(): MutableLiveData<String> {
        return body
    }

    fun onClickPost() {
        // Initialize new activity from here, perhaps?
    }
}

Und im Layout-XML wird ein onClick-Attribut festgelegt

Android: onClick = "@ {() -> viewModel.onClickPost ()}"

auf diese onClickPost-Methode zu zeigen funktioniert zwar, aber ich kann die Intent nicht von dort aus initialisieren. Ich habe viele Möglichkeiten ausprobiert, um den Kontext der MainActivitiy ohne Erfolg zu ermitteln, wie z

val intent = Intent (MainActivity :: getApplicationContext, PostDetailActivity :: class.Java)

Es zeigt jedoch rechtzeitig einen Fehler an.

3
gamofe

Versuchen Sie es mit einem SingleLiveEvent

Hier ist der Code dafür aus Googles architecture samples repo (falls es jemals aus dem Repo entfernt wird):

import Android.Arch.lifecycle.LifecycleOwner;
import Android.Arch.lifecycle.MutableLiveData;
import Android.Arch.lifecycle.Observer;
import Android.support.annotation.MainThread;
import Android.support.annotation.Nullable;
import Android.util.Log;

import Java.util.concurrent.atomic.AtomicBoolean;

/**
 * A lifecycle-aware observable that sends only new updates after subscription, used for events like
 * navigation and Snackbar messages.
 * <p>
 * This avoids a common problem with events: on configuration change (like rotation) an update
 * can be emitted if the observer is active. This LiveData only calls the observable if there's an
 * explicit call to setValue() or call().
 * <p>
 * Note that only one observer is going to be notified of changes.
 */
public class SingleLiveEvent<T> extends MutableLiveData<T> {

    private static final String TAG = "SingleLiveEvent";

    private final AtomicBoolean mPending = new AtomicBoolean(false);

    @MainThread
    public void observe(LifecycleOwner owner, final Observer<T> observer) {

        if (hasActiveObservers()) {
            Log.w(TAG, "Multiple observers registered but only one will be notified of changes.");
        }

        // Observe the internal MutableLiveData
        super.observe(owner, new Observer<T>() {
            @Override
            public void onChanged(@Nullable T t) {
                if (mPending.compareAndSet(true, false)) {
                    observer.onChanged(t);
                }
            }
        });
    }

    @MainThread
    public void setValue(@Nullable T t) {
        mPending.set(true);
        super.setValue(t);
    }

    /**
     * Used for cases where T is Void, to make calls cleaner.
     */
    @MainThread
    public void call() {
        setValue(null);
    }
}
1
MidasLefko

Versuchen Sie: Android:onClick="@{(view) -> viewModel.onClickPost(view)}"

Ändern Sie auch onClickPost, um eine Ansicht zu übernehmen. Dann können Sie die view.getContext()-Methode für die Ansicht verwenden, um auf den in dieser Ansicht gespeicherten Kontext zuzugreifen. 

Da ViewModels jedoch nicht auf eine Ansicht oder eine andere Klasse verweisen soll, die den Kontext einer Aktivität enthält, ist es unangebracht, Ihre Logik für das Starten einer Aktivität in ViewModel zu platzieren. Sie sollten auf jeden Fall einen separaten Ort dafür in Betracht ziehen. 

Persönlich, für meinen Code, wenn es sich um eine einfache startActivity ohne zusätzliches Gepäck handelt, erstelle ich eine separate Klasse, die eine statische Methode enthält. Durch das Datenbinden importiere ich diese Klasse und verwende sie im onClick, um eine neue Aktivität mit der oben genannten Methode zu starten. 

Ein Beispiel dafür: 

public class ActivityHandler{        
    public static void showNextActivity(View view, ViewModel viewModel){
        Intent intent = new Intent(); //Create your intent and add extras if needed
        view.getContext().startActivity(intent);
    }
}

<layout xmlns:Android="http://schemas.Android.com/apk/res/Android">
    <data>
        <import type="whatever.you.want.ActivityHandler" />
        <variable name="viewmodel" type="whatever.you.want.here.too.ViewModel" />
    </data>

    <Button
        //Regular layout properties
        Android:onClick="@{(view) -> ActivityHandler.showNextActivity(view, viewmodel)}"
        />
</layout>

Sehen Sie sich Listener Bindings hier an: https://developer.Android.com/topic/libraries/data-binding/expressions#listener_bindings

Abhängig von der erforderlichen Datenmenge können Sie Ihren startActivity-Code jedoch in anderen Klassen platzieren, die am besten zum Design Ihrer App passen.

1
Jackey

Sie können sogar eine Aktivitätsinstanz an das Modell oder das Layout übergeben, aber das würde ich nicht vorziehen.

Die bevorzugte Methode ist, die Schnittstelle an das Zeilenlayout zu übergeben.

variable in Layoutdaten deklarieren

<variable
    name="onClickListener"
    type="Android.view.View.OnClickListener"/>

rufen Sie dies auf, wenn Sie darauf klicken

<LinearLayout
    Android:layout_width="match_parent"
    Android:layout_height="wrap_content"
    Android:onClick="@{onClickListener::onClick}"
    >

setze auch diesen Listener vom Adapter

 binding.viewModel = viewModel
 binding.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {
            context.startActivity(new Intent(context, MainActivity.class));
        }
    });
1
Khemraj