web-dev-qa-db-fra.com

Obtenir le contexte d'une activité à partir de la classe de modèle de vue

J'ai basé mon code sur un exemple utilisant des composants d'architecture Android et la liaison de données. C'est une nouvelle façon pour moi, et la façon dont c'est codé rend difficile l'ouverture correcte d'une nouvelle activité avec les informations de la publication sur laquelle l'utilisateur a cliqué.

C'est l'adaptateur des posts

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
        }
    }
}

La méthode bind provient de la classe de modèle de vue:

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?
    }
}

Et dans la mise en page XML, définir un attribut onClick

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

pointer vers cette méthode onClickPost fonctionne, mais je ne peux pas initialiser le Intent à partir de là. J’ai essayé de nombreuses manières d’acquérir le contexte de MainActivitiy, sans succès, tel que

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

Mais il affiche une erreur à l'heure.

3
gamofe

Essayez d’utiliser un SingleLiveEvent

Voici le code correspondant à partir de exemples d'architecture Googles repo (au cas où il serait retiré du rapport):

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

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

Modifiez également onClickPost pour intégrer une vue. Ensuite, vous pouvez utiliser la méthode view.getContext() sur la vue pour accéder au contexte stocké dans cette vue. 

Cependant, comme ViewModels ne devrait pas faire référence à une vue ou à une autre classe contenant le contexte d'une activité, il est tout à fait inapproprié de placer votre logique de démarrage d'une activité dans ViewModel. Vous devriez certainement envisager un endroit séparé pour le faire. 

Personnellement, pour mon code, s'il s'agit d'un simple startActivity sans bagage supplémentaire, je crée une classe distincte qui contient une méthode statique. Grâce à la liaison de données, je vais importer cette classe et l'utiliser dans onClick pour démarrer une nouvelle activité en utilisant la méthode que j'ai décrite ci-dessus. 

Un exemple de ceci: 

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>

Regardez les liens d'écoute ici: https://developer.Android.com/topic/libraries/data-binding/expressions#listener_bindings

Toutefois, en fonction de la quantité de données nécessaire, vous pouvez placer votre code startActivity dans d'autres classes qui correspondent le mieux à la conception de votre application.

1
Jackey

Vous pouvez même transmettre l'instance d'activité au modèle ou à la présentation, mais je ne préférerai pas cela.

Le moyen préféré est de passer d’une interface à la disposition des lignes.

déclarer une variable dans les données de mise en page

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

invoquer cela lorsque vous cliquez dessus

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

définir également cet auditeur de l'adaptateur

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