
org.droitateddb.cursor.CombinedCursorImpl Maven / Gradle / Ivy
/*
* Copyright (C) 2014 The droitated DB Authors
*
* 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 org.droitateddb.cursor;
import android.content.Context;
import android.database.Cursor;
import org.droitateddb.DbCreator;
import org.droitateddb.schema.AbstractAttribute;
import org.droitateddb.schema.EntityInfo;
import org.droitateddb.schema.SchemaConstants;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Iterator;
import java.util.NoSuchElementException;
import java.util.concurrent.atomic.AtomicBoolean;
import static org.droitateddb.Utilities.getDeclaredField;
import static org.droitateddb.Utilities.getStaticFieldValue;
import static org.droitateddb.Utilities.handle;
/**
* @param Entity type represented within the Cursor
* @author Falk Appel
* @author Alexander Frank
*/
public class CombinedCursorImpl extends ProxyableCursor implements CombinedCursor {
private final AbstractAttribute[] attributes;
private final Class entityClass;
private final Cursor originalCursor;
private AtomicBoolean closed = new AtomicBoolean(false);
private CombinedCursorImpl(final Cursor originalCursor, final Class entityClass, final AbstractAttribute[] attributes) {
this.originalCursor = originalCursor;
this.entityClass = entityClass;
this.attributes = attributes;
}
@SuppressWarnings("unchecked")
public static final CombinedCursor create(Context context, final Cursor originalCursor, final EntityInfo entityInfo, final Class entityClass) {
try {
Class> definition = entityInfo.definition();
AbstractAttribute[] attributes = getStaticFieldValue(definition, SchemaConstants.ATTRIBUTES);
final Context appContext = context.getApplicationContext();
final CombinedCursorImpl magicCursor = new CombinedCursorImpl(originalCursor, entityClass, attributes);
InvocationHandler handler = new InvocationHandler() {
@Override
public Object invoke(final Object proxy, final Method method, final Object[] args) throws Throwable {
Class>[] argTypes;
if (args == null) {
argTypes = new Class>[0];
} else {
argTypes = ReflectionUtil.getArgTypes(args);
}
try {
if (ReflectionUtil.isCloseMethod(method)) {
method.invoke(originalCursor, args);
if (magicCursor.closed.compareAndSet(false, true)) {
DbCreator.getInstance(appContext).reduceDatabaseConnection();
}
return null;
} else if (ReflectionUtil.isMethodOfType(method, argTypes, Cursor.class)) {
return method.invoke(originalCursor, args);
} else if (ReflectionUtil.isMethodOfType(method, argTypes, ObjectCursor.class)) {
return method.invoke(magicCursor, args);
} else {
throw new UnsupportedOperationException("Unexpected method call for " + method.toString());
}
} catch (InvocationTargetException ite) {
throw ite.getTargetException();
}
}
};
return (CombinedCursor) Proxy.newProxyInstance(CombinedCursorImpl.class.getClassLoader(), new Class>[]{CombinedCursor.class}, handler);
} catch (Exception e) {
throw handle(e);
}
}
private void assertEmptyCursor() {
if (originalCursor.getCount() == 0) {
throw new NoSuchElementException();
}
}
private T construct() {
try {
Constructor constructor = entityClass.getConstructor();
T instance = constructor.newInstance();
for (AbstractAttribute attribute : attributes) {
Field field = getDeclaredField(entityClass, attribute.fieldName());
field.setAccessible(true);
field.set(instance, attribute.getValueFromCursor(originalCursor));
}
return instance;
} catch (Exception e) {
throw handle(e);
}
}
@Override
public Collection getAll() {
moveBeforeFirst();
Collection instances = new ArrayList(originalCursor.getCount());
while (originalCursor.moveToNext()) {
instances.add(construct());
}
return instances;
}
private void moveBeforeFirst() {
originalCursor.moveToFirst();
originalCursor.moveToPrevious();
}
@Override
public T getCurrent() {
assertEmptyCursor();
if (originalCursor.isBeforeFirst()) {
originalCursor.moveToFirst();
}
if (originalCursor.isAfterLast()) {
originalCursor.moveToLast();
}
return construct();
}
@Override
public T getFirst() {
assertEmptyCursor();
originalCursor.moveToFirst();
return construct();
}
@Override
public T getLast() {
assertEmptyCursor();
originalCursor.moveToLast();
return construct();
}
@Override
public T getNext() {
assertEmptyCursor();
if (!hasNext()) {
throw new NoSuchElementException("There is no next element in cursor!");
}
originalCursor.moveToNext();
return construct();
}
@Override
public Collection getNext(final int amount) {
assertEmptyCursor();
if (!hasNext()) {
throw new NoSuchElementException("There is no next element in cursor!");
}
Collection instances = new ArrayList(originalCursor.getCount());
int count = 0;
while (originalCursor.moveToNext() && count < amount) {
if (!originalCursor.isAfterLast()) {
instances.add(construct());
count++;
}
}
return instances;
}
@Override
public T getOne() {
assertEmptyCursor();
if (originalCursor.getCount() > 1) {
throw new IllegalStateException("Expected only one element in cursor but there were " + originalCursor.getCount() + "!");
}
originalCursor.moveToFirst();
return construct();
}
@Override
public T getPrevious() {
assertEmptyCursor();
if (!hasPrevious()) {
throw new NoSuchElementException("There is no previos element in cursor!");
}
originalCursor.moveToPrevious();
return construct();
}
@Override
public Collection getPrevious(final int amount) {
assertEmptyCursor();
if (!hasPrevious()) {
throw new NoSuchElementException("There is no next element in cursor!");
}
Collection instances = new ArrayList(originalCursor.getCount());
int count = 0;
while (originalCursor.moveToPrevious() && count < amount) {
if (!originalCursor.isBeforeFirst()) {
instances.add(construct());
count++;
}
}
return instances;
}
@Override
public boolean hasNext() {
return originalCursor.getCount() > 0 && !(originalCursor.isLast() || originalCursor.isAfterLast());
}
@Override
public boolean hasPrevious() {
return originalCursor.getCount() > 0 && !(originalCursor.isFirst() || originalCursor.isBeforeFirst());
}
@Override
public Iterator iterator() {
moveBeforeFirst();
return new Iterator() {
@Override
public boolean hasNext() {
return CombinedCursorImpl.this.hasNext();
}
@Override
public T next() {
return getNext();
}
@Override
public void remove() {
throw new UnsupportedOperationException();
}
};
}
@Override
public int size() {
return originalCursor.getCount();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy