Upgrade to Pro — share decks privately, control downloads, hide ads and more …

ViewModel: A deep dive (GDG Heraklion)

ViewModel: A deep dive (GDG Heraklion)

At Google I/O we got the Android architecture components.
One of its most interesting concepts is the new ViewModel.
How does it work? How do I plug it into my app? Can I use it without any of the other components?
Can I plug it into an existing MVVM or how would I build MVVM with Google's ViewModel?

Danny Preussler

July 27, 2017
Tweet

More Decks by Danny Preussler

Other Decks in Technology

Transcript

  1. ViewModel in data binding <TextView … android:id="@+id/all_shows_item_title" android:text="@{viewModel.title}" /> <android.support.v7.widget.CardView

    … android:onClick="@{() -> viewModel.onClicked()}"> <data> <variable name="viewModel" type="com.vmn.playplex.main.allshows.SeriesViewModel"/> </data>
  2. Is MVP dead? It’s like Java and Kotlin: •MVP will

    stay for quite some time •There is a new cooler kid in town that won’t leave Put the ViewModel behind Presenter
  3. Is MVP dead? It’s like Java and Kotlin: •MVP will

    stay for quite some time •There is a new cooler kid in town that won’t leave Put the ViewModel behind Presenter
  4. Is MVP dead? It’s like Java and Kotlin: •MVP will

    stay for quite some time •There is a new cooler kid in town that won’t leave Put the ViewModel behind Presenter
  5. Is MVP dead? It’s like Java and Kotlin: •MVP will

    stay for quite some time •There is a new cooler kid in town that won’t leave Put the ViewModel behind Presenter
  6. In architecture components •A ViewModel provides the data for a

    specific UI •The ViewModel does not know about the View! •Survives configuration change
  7. In architecture components •A ViewModel provides the data for a

    specific UI •The ViewModel does not know about the View! •Survives configuration change
  8. In architecture components •A ViewModel provides the data for a

    specific UI •The ViewModel does not know about the View! •Survives configuration change
  9. How to use class MyViewModel extends ObservableViewModel { Coming soon

    Jose Alcérreca, Google https://medium.com/@dpreussler/add-the-new-viewmodel-to-your-mvvm-36bfea86b159
  10. How to use public void onStopped() { … } No

    more life cycle forwarding!
  11. How to use class MyViewModelFactory implements ViewModelProvider.Factory { @Inject MyUseCase

    useCase; @Override public MyViewModel create(Class modelClass) { return new MyViewModel(useCase); } }
  12. How to use •Always try to build your own Factory

    •Default factory uses newInstance() which is some hundred times slower than new calls (reflection) https://speakerdeck.com/dpreussler/comparing-dependency-injection- frameworks-and-internals-for-android
  13. How to use •Always try to build your own Factory

    •Default factory uses newInstance() which is some hundred times slower than new calls (reflection) https://speakerdeck.com/dpreussler/comparing-dependency-injection- frameworks-and-internals-for-android
  14. What if… class MyViewModel extends ViewModel { @Override protected void

    onCleared() { super.onCleared(); cleanupSubscriptions(); }
  15. How does it actually work? class HolderFragment extends Fragment {

    public HolderFragment() { setRetainInstance(true); } …
  16. How does it actually work? @Override public void onDestroy() {

    super.onDestroy(); mViewModelStore.clear(); }
  17. Could you write that? •What if asked for ViewModel but

    fragment transaction not done yet? •You might end up with duplicates
  18. Could you write that? •What if asked for ViewModel but

    fragment transaction not done yet? •You might end up with duplicates
  19. Could you write that? static class HolderFragmentManager { private Map<Activity,

    HolderFragment> mNotCommittedActivityHolders = new HashMap<>(); private Map<Fragment, HolderFragment> mNotCommittedFragmentHolders = new HashMap<>(); private ActivityLifecycleCallbacks mActivityCallbacks = new EmptyActivityLifecycleCallbacks() { @Override public void onActivityDestroyed(Activity activity) { HolderFragment fragment = mNotCommittedActivityHolders.remove(activity); …
  20. Could you write that? •What if activity dies before fragment

    transaction? •You might end up with memory leaks
  21. Could you write that? private ActivityLifecycleCallbacks mActivityCallbacks = new EmptyActivityLifecycleCallbacks()

    { @Override public void onActivityDestroyed(Activity activity) { HolderFragment fragment = mNotCommittedActivityHolders.remove(activity); if (fragment != null) { Log.e(LOG_TAG, "Failed to save a ViewModel for " + activity); } } }; …
  22. What if Two Fragments of same Activity ask for same

    ViewModel.class via ViewModelProviders .of(this) .get(MyViewModel.class);
  23. What if Two Fragments of same Activity ask for same

    ViewModel.class via ViewModelProviders .of(this) .get(MyViewModel.class);
  24. result •Fragment and Activity share the same FragmentManager •But implementation

    uses Activity’s FragmentManager but ChildFragmentManager for Fragments
  25. result •Fragment and Activity share the same FragmentManager •But implementation

    uses Activity’s FragmentManager but ChildFragmentManager for Fragments
  26. What if Two Fragments of same Activity ask for same

    ViewModel.class via ViewModelProviders .of(getActivity()) .get(MyViewModel.class);
  27. all problems solved? ViewModels provide a convenient way to retain

    data across configuration changes but they are not persisted if the application is killed by the operating system https://developer.android.com/topic/libraries/architecture/viewmodel.html#viewm odel_vs_savedinstancestate
  28. Why? The data saved via onSaveInstanceState is kept in the

    system process memory and the Android OS allows you to keep only a very small amount of data so it is not a good place to keep actual data for your app. TransactionTooLargeException anyone?
  29. after The truth •Keep non-UI states in non-UI layer Not

    in bundle! •Use real caching strategies •Allows updating cache in background
  30. after The truth •Keep non-UI states in non-UI layer Not

    in bundle! •Use real caching strategies •Allows updating cache in background
  31. after The truth •Keep non-UI states in non-UI layer Not

    in bundle! •Use real caching strategies •Allows updating cache in background
  32. lets tweak it class MyModelFactory implements ViewModelProvider.Factory { … public

    MyModelFactory(Bundle bundle) { this.bundle = bundle; } @Override public MyViewModel create(Class modelClass) { MyViewModel viewModel = new MyViewModel(); viewModel.readFrom(bundle); return viewModel; }
  33. list item ViewModels In MVVM: •each item in RecyclerView should

    be backed by a ViewModel •ListViewModel and ItemViewModels •ItemViewModels normally don’t retrieve their own state so no need to extend Googles ViewModel
  34. list item ViewModels In MVVM: •each item in RecyclerView should

    be backed by a ViewModel •ListViewModel and ItemViewModels •ItemViewModels normally don’t retrieve their own state so no need to extend Googles ViewModel
  35. list item ViewModels In MVVM: •each item in RecyclerView should

    be backed by a ViewModel •ListViewModel and ItemViewModels •ItemViewModels normally don’t retrieve their own state so no need to extend Googles ViewModel
  36. list item ViewModels •Key could be position? public <T extends

    ViewModel> T get( String key, Class<T> modelClass)
  37. list item ViewModels “ViewModelProvider, which will create ViewModels via the

    given Factory and retain them in a ViewModelStore “
  38. public class ViewModelStore { private final HashMap<String, ViewModel> mMap =

    new HashMap<>(); final void put(String key, ViewModel viewModel) {…} final ViewModel get(String key) { …} public final void clear() {…} }
  39. public class ViewModelStore { private final HashMap<String, ViewModel> mMap =

    new HashMap<>(); final void put(String key, ViewModel viewModel) {…} final ViewModel get(String key) { …} public final void clear() {…} }
  40. Can I do it differently? •Let’s assume we build transactional

    flow spreading over multiple Activities and wand to keep data in ViewModel •Can not use the retained fragment approach •Need to override how the ViewModel is stored •Need to override the ViewModelStore is stored
  41. Viewmodelstoreowner “A responsibility of an implementation of this interface is

    to retain owned ViewModelStore during the configuration changes and call ViewModelStore#clear(), when this scope is going to be destroyed.“
  42. ViewModelproviders public static ViewModelProvider of( Fragment fragment, Factory factory) {

    return new ViewModelProvider( ViewModelStores.of(fragment), factory); }
  43. How to extend life? Toothpick: A scope tree based Dependency

    Injection (DI) library https://github.com/stephanenicolas/toothpick
  44. lets do it class LongLivingViewModelStoreOwner implements ViewModelStoreOwner { @Override public

    ViewModelStore getViewModelStore() { return Toothpick.openScope(SCOPE_NAME) .getInstance(ViewModelStore.class); } public void discardViewModel() { Toothpick.closeScope(SCOPE_NAME); }
  45. let`s sum up •Well designed API •Know the life time:

    Never have Activivity or View reference in ViewModel •Be aware of Recreation •Careful with RecyclerView •Still alpha