ObservableMapAdapter: Based on an observable TreeMap
Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
This commit is contained in:
parent
5023c937ad
commit
97149fff3f
@ -0,0 +1,111 @@
|
||||
package com.wireguard.android;
|
||||
|
||||
import android.content.Context;
|
||||
import android.databinding.DataBindingUtil;
|
||||
import android.databinding.ObservableMap;
|
||||
import android.databinding.ViewDataBinding;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.BaseAdapter;
|
||||
import android.widget.ListAdapter;
|
||||
|
||||
import java.lang.ref.WeakReference;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
|
||||
/**
|
||||
* A generic ListAdapter backed by a TreeMap that adds observability.
|
||||
*/
|
||||
|
||||
class ObservableMapAdapter<K extends Comparable<K>, V> extends BaseAdapter implements ListAdapter {
|
||||
private ArrayList<K> keys;
|
||||
private final int layoutId;
|
||||
private final LayoutInflater layoutInflater;
|
||||
private ObservableSortedMap<K, V> map;
|
||||
private final OnMapChangedCallback<K, V> callback = new OnMapChangedCallback<>(this);
|
||||
|
||||
ObservableMapAdapter(final Context context, final int layoutId,
|
||||
final ObservableSortedMap<K, V> map) {
|
||||
layoutInflater = LayoutInflater.from(context);
|
||||
this.layoutId = layoutId;
|
||||
setMap(map);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getCount() {
|
||||
return map != null ? map.size() : 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public V getItem(final int position) {
|
||||
if (map == null || position < 0 || position >= map.size())
|
||||
return null;
|
||||
return map.get(getKeys().get(position));
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getItemId(final int position) {
|
||||
if (map == null || position < 0 || position >= map.size())
|
||||
return -1;
|
||||
return getKeys().get(position).hashCode();
|
||||
}
|
||||
|
||||
public int getItemPosition(final K key) {
|
||||
if (map == null)
|
||||
return -1;
|
||||
return Collections.binarySearch(getKeys(), key);
|
||||
}
|
||||
|
||||
private ArrayList<K> getKeys() {
|
||||
if (keys == null)
|
||||
keys = new ArrayList<>(map.keySet());
|
||||
return keys;
|
||||
}
|
||||
|
||||
@Override
|
||||
public View getView(final int position, final View convertView, final ViewGroup parent) {
|
||||
ViewDataBinding binding = DataBindingUtil.getBinding(convertView);
|
||||
if (binding == null)
|
||||
binding = DataBindingUtil.inflate(layoutInflater, layoutId, parent, false);
|
||||
binding.setVariable(BR.item, getItem(position));
|
||||
binding.executePendingBindings();
|
||||
return binding.getRoot();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean hasStableIds() {
|
||||
return true;
|
||||
}
|
||||
|
||||
public void setMap(final ObservableSortedMap<K, V> newMap) {
|
||||
if (map != null)
|
||||
map.removeOnMapChangedCallback(callback);
|
||||
keys = null;
|
||||
map = newMap;
|
||||
if (map != null) {
|
||||
map.addOnMapChangedCallback(callback);
|
||||
}
|
||||
}
|
||||
|
||||
private static class OnMapChangedCallback<K extends Comparable<K>, V>
|
||||
extends ObservableMap.OnMapChangedCallback<ObservableSortedMap<K, V>, K, V> {
|
||||
|
||||
private final WeakReference<ObservableMapAdapter<K, V>> weakAdapter;
|
||||
|
||||
private OnMapChangedCallback(final ObservableMapAdapter<K, V> adapter) {
|
||||
weakAdapter = new WeakReference<>(adapter);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onMapChanged(final ObservableSortedMap<K, V> sender, final K key) {
|
||||
final ObservableMapAdapter<K, V> adapter = weakAdapter.get();
|
||||
if (adapter != null) {
|
||||
adapter.keys = null;
|
||||
adapter.notifyDataSetChanged();
|
||||
} else {
|
||||
sender.removeOnMapChangedCallback(this);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,12 @@
|
||||
package com.wireguard.android;
|
||||
|
||||
import android.databinding.ObservableMap;
|
||||
|
||||
import java.util.SortedMap;
|
||||
|
||||
/**
|
||||
* Interface for maps that are both observable and sorted.
|
||||
*/
|
||||
|
||||
public interface ObservableSortedMap<K, V> extends ObservableMap<K, V>, SortedMap<K, V> {
|
||||
}
|
@ -0,0 +1,67 @@
|
||||
package com.wireguard.android;
|
||||
|
||||
import android.databinding.MapChangeRegistry;
|
||||
import android.databinding.ObservableMap;
|
||||
import android.support.annotation.NonNull;
|
||||
|
||||
import java.util.Iterator;
|
||||
import java.util.Map;
|
||||
import java.util.TreeMap;
|
||||
|
||||
/**
|
||||
* Observable version of a TreeMap. Only notifies for changes made through methods, not iterators or
|
||||
* views. This behavior is in line with that of ObservableArrayMap.
|
||||
*/
|
||||
|
||||
public class ObservableTreeMap<K, V> extends TreeMap<K, V> implements ObservableSortedMap<K, V> {
|
||||
private transient MapChangeRegistry listeners;
|
||||
|
||||
@Override
|
||||
public void clear() {
|
||||
super.clear();
|
||||
notifyChange(null);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void addOnMapChangedCallback(
|
||||
final OnMapChangedCallback<? extends ObservableMap<K, V>, K, V> listener) {
|
||||
if (listeners == null)
|
||||
listeners = new MapChangeRegistry();
|
||||
listeners.add(listener);
|
||||
}
|
||||
|
||||
private void notifyChange(final K key) {
|
||||
if (listeners != null)
|
||||
listeners.notifyChange(this, key);
|
||||
}
|
||||
|
||||
@Override
|
||||
public V put(final K key, final V value) {
|
||||
final V oldValue = super.put(key, value);
|
||||
notifyChange(key);
|
||||
return oldValue;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void putAll(@NonNull final Map<? extends K, ? extends V> map) {
|
||||
super.putAll(map);
|
||||
for (final K key : map.keySet())
|
||||
notifyChange(key);
|
||||
}
|
||||
|
||||
@Override
|
||||
public V remove(final Object key) {
|
||||
final V oldValue = super.remove(key);
|
||||
@SuppressWarnings("unchecked")
|
||||
final K k = (K) key;
|
||||
notifyChange(k);
|
||||
return oldValue;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void removeOnMapChangedCallback(
|
||||
final OnMapChangedCallback<? extends ObservableMap<K, V>, K, V> listener) {
|
||||
if (listeners != null)
|
||||
listeners.remove(listener);
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user