package eu.faircode.email; /* This file is part of FairEmail. FairEmail is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. FairEmail is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with FairEmail. If not, see . Copyright 2018-2019 by Marcel Bokhorst (M66B) */ import java.util.HashMap; import java.util.Map; import androidx.lifecycle.Lifecycle; import androidx.lifecycle.LifecycleObserver; import androidx.lifecycle.LifecycleOwner; import androidx.lifecycle.LiveData; import androidx.lifecycle.Observer; import androidx.lifecycle.OnLifecycleEvent; import androidx.lifecycle.ViewModel; import androidx.paging.PagedList; public class ViewModelMessages extends ViewModel { private Map>> messages = new HashMap<>(); void setMessages(AdapterMessage.ViewType viewType, LifecycleOwner owner, final LiveData> messages) { if (viewType == AdapterMessage.ViewType.UNIFIED) viewType = AdapterMessage.ViewType.FOLDER; this.messages.put(viewType, messages); if (viewType == AdapterMessage.ViewType.FOLDER) this.messages.remove(AdapterMessage.ViewType.SEARCH); if (viewType == AdapterMessage.ViewType.THREAD) owner.getLifecycle().addObserver(new LifecycleObserver() { @OnLifecycleEvent(Lifecycle.Event.ON_DESTROY) public void onDestroyed() { Log.i("Removed model thread"); ViewModelMessages.this.messages.remove(AdapterMessage.ViewType.THREAD); } }); else { // Keep list up-to-date for previous/next navigation messages.observeForever(new Observer>() { @Override public void onChanged(PagedList messages) { } }); } } void observe(AdapterMessage.ViewType viewType, LifecycleOwner owner, Observer> observer) { if (viewType == AdapterMessage.ViewType.UNIFIED) viewType = AdapterMessage.ViewType.FOLDER; if (owner.getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.INITIALIZED)) messages.get(viewType).observe(owner, observer); } void removeObservers(AdapterMessage.ViewType viewType, LifecycleOwner owner) { if (viewType == AdapterMessage.ViewType.UNIFIED) viewType = AdapterMessage.ViewType.FOLDER; if (owner.getLifecycle().getCurrentState().isAtLeast(Lifecycle.State.INITIALIZED)) { LiveData> list = messages.get(viewType); if (list != null) list.removeObservers(owner); } } @Override protected void onCleared() { messages.clear(); } void observePrevNext(LifecycleOwner owner, final long id, boolean found, final IPrevNext intf) { AdapterMessage.ViewType viewType = (found ? AdapterMessage.ViewType.SEARCH : AdapterMessage.ViewType.FOLDER); LiveData> list = messages.get(viewType); if (list == null) { Log.w("Observe previous/next without list"); return; } Log.i("Observe previous/next id=" + id); list.observe(owner, new Observer>() { @Override public void onChanged(PagedList messages) { Log.i("Observe previous/next id=" + id + " messages=" + messages.size()); for (int pos = 0; pos < messages.size(); pos++) { TupleMessageEx item = messages.get(pos); if (item != null && id == item.id) { boolean load = false; if (pos - 1 >= 0) { TupleMessageEx next = messages.get(pos - 1); if (next == null) load = true; intf.onNext(true, next == null ? null : next.id); } else intf.onNext(false, null); if (pos + 1 < messages.size()) { TupleMessageEx prev = messages.get(pos + 1); if (prev == null) load = true; intf.onPrevious(true, prev == null ? null : prev.id); } else intf.onPrevious(false, null); intf.onFound(pos, messages.size()); if (load) messages.loadAround(pos); return; } } Log.w("Observe previous/next gone id=" + id); } }); } interface IPrevNext { void onPrevious(boolean exists, Long id); void onNext(boolean exists, Long id); void onFound(int position, int size); } }