blob: 9a83b5bd83289d6c8eea4d895101ae1722d66c03 [file] [log] [blame]
/*
* Copyright (C) 2019 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.android.systemui.classifier.brightline;
import android.view.MotionEvent;
import java.util.Collection;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.ListIterator;
/**
* Maintains an ordered list of the last N milliseconds of MotionEvents.
*
* This class is simply a convenience class designed to look like a simple list, but that
* automatically discards old MotionEvents. It functions much like a queue - first in first out -
* but does not have a fixed size like a circular buffer.
*/
public class TimeLimitedMotionEventBuffer implements List<MotionEvent> {
private final LinkedList<MotionEvent> mMotionEvents;
private long mMaxAgeMs;
TimeLimitedMotionEventBuffer(long maxAgeMs) {
super();
this.mMaxAgeMs = maxAgeMs;
this.mMotionEvents = new LinkedList<>();
}
private void ejectOldEvents() {
if (mMotionEvents.isEmpty()) {
return;
}
Iterator<MotionEvent> iter = listIterator();
long mostRecentMs = mMotionEvents.getLast().getEventTime();
while (iter.hasNext()) {
MotionEvent ev = iter.next();
if (mostRecentMs - ev.getEventTime() > mMaxAgeMs) {
iter.remove();
ev.recycle();
}
}
}
@Override
public void add(int index, MotionEvent element) {
throw new UnsupportedOperationException();
}
@Override
public MotionEvent remove(int index) {
return mMotionEvents.remove(index);
}
@Override
public int indexOf(Object o) {
return mMotionEvents.indexOf(o);
}
@Override
public int lastIndexOf(Object o) {
return mMotionEvents.lastIndexOf(o);
}
@Override
public int size() {
return mMotionEvents.size();
}
@Override
public boolean isEmpty() {
return mMotionEvents.isEmpty();
}
@Override
public boolean contains(Object o) {
return mMotionEvents.contains(o);
}
@Override
public Iterator<MotionEvent> iterator() {
return mMotionEvents.iterator();
}
@Override
public Object[] toArray() {
return mMotionEvents.toArray();
}
@Override
public <T> T[] toArray(T[] a) {
return mMotionEvents.toArray(a);
}
@Override
public boolean add(MotionEvent element) {
boolean result = mMotionEvents.add(element);
ejectOldEvents();
return result;
}
@Override
public boolean remove(Object o) {
return mMotionEvents.remove(o);
}
@Override
public boolean containsAll(Collection<?> c) {
return mMotionEvents.containsAll(c);
}
@Override
public boolean addAll(Collection<? extends MotionEvent> collection) {
boolean result = mMotionEvents.addAll(collection);
ejectOldEvents();
return result;
}
@Override
public boolean addAll(int index, Collection<? extends MotionEvent> elements) {
throw new UnsupportedOperationException();
}
@Override
public boolean removeAll(Collection<?> c) {
return mMotionEvents.removeAll(c);
}
@Override
public boolean retainAll(Collection<?> c) {
return mMotionEvents.retainAll(c);
}
@Override
public void clear() {
mMotionEvents.clear();
}
@Override
public boolean equals(Object o) {
return mMotionEvents.equals(o);
}
@Override
public int hashCode() {
return mMotionEvents.hashCode();
}
@Override
public MotionEvent get(int index) {
return mMotionEvents.get(index);
}
@Override
public MotionEvent set(int index, MotionEvent element) {
throw new UnsupportedOperationException();
}
@Override
public ListIterator<MotionEvent> listIterator() {
return new Iter(0);
}
@Override
public ListIterator<MotionEvent> listIterator(int index) {
return new Iter(index);
}
@Override
public List<MotionEvent> subList(int fromIndex, int toIndex) {
throw new UnsupportedOperationException();
}
class Iter implements ListIterator<MotionEvent> {
private final ListIterator<MotionEvent> mIterator;
Iter(int index) {
this.mIterator = mMotionEvents.listIterator(index);
}
@Override
public boolean hasNext() {
return mIterator.hasNext();
}
@Override
public MotionEvent next() {
return mIterator.next();
}
@Override
public boolean hasPrevious() {
return mIterator.hasPrevious();
}
@Override
public MotionEvent previous() {
return mIterator.previous();
}
@Override
public int nextIndex() {
return mIterator.nextIndex();
}
@Override
public int previousIndex() {
return mIterator.previousIndex();
}
@Override
public void remove() {
mIterator.remove();
}
@Override
public void set(MotionEvent motionEvent) {
throw new UnsupportedOperationException();
}
@Override
public void add(MotionEvent element) {
throw new UnsupportedOperationException();
}
}
}