All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.droitateddb.cursor.CombinedCursorImpl Maven / Gradle / Ivy

Go to download

droitatedDB is a lightweight framework, which frees you from the burden of dealing with Androids SQLite database directly if you don't want to and let's you access it directly if you have to.

The newest version!
/*
 * 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 - 2024 Weber Informatics LLC | Privacy Policy