com.abubusoft.kripton.processor.sqlite.BindAsyncTaskBuilder Maven / Gradle / Ivy
/*******************************************************************************
* Copyright 2015, 2017 Francesco Benincasa ([email protected]).
*
* 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.abubusoft.kripton.processor.sqlite;
import static com.abubusoft.kripton.processor.core.reflect.TypeUtility.className;
import java.io.IOException;
import java.util.concurrent.Executor;
import javax.annotation.processing.Filer;
import javax.lang.model.element.Modifier;
import javax.lang.model.element.PackageElement;
import javax.lang.model.util.Elements;
import com.abubusoft.kripton.android.BindAsyncTaskType;
import com.abubusoft.kripton.android.Logger;
import com.abubusoft.kripton.android.annotation.BindDataSource;
import com.abubusoft.kripton.processor.bind.JavaWriterHelper;
import com.abubusoft.kripton.processor.core.reflect.TypeUtility;
import com.abubusoft.kripton.processor.sqlite.core.JavadocUtility;
import com.abubusoft.kripton.processor.sqlite.model.SQLiteDatabaseSchema;
import com.abubusoft.kripton.processor.utils.AnnotationProcessorUtilis;
import com.squareup.javapoet.AnnotationSpec;
import com.squareup.javapoet.ArrayTypeName;
import com.squareup.javapoet.FieldSpec;
import com.squareup.javapoet.MethodSpec;
import com.squareup.javapoet.ParameterSpec;
import com.squareup.javapoet.ParameterizedTypeName;
import com.squareup.javapoet.TypeSpec;
import com.squareup.javapoet.TypeSpec.Builder;
import com.squareup.javapoet.TypeVariableName;
import android.os.AsyncTask;
/**
* Utility class to generate async task for database operations.
*/
public class BindAsyncTaskBuilder {
/** The Constant PREFIX. */
public static final String PREFIX = "Bind";
/** The Constant SUFFIX. */
public static final String SUFFIX = "AsyncTask";
/** The builder. */
private static Builder builder;
/**
* Generate async task for database operations.
*
* @param elementUtils the element utils
* @param filer the filer
* @param schema the schema
* @throws IOException Signals that an I/O exception has occurred.
*/
public static void generate(Elements elementUtils, Filer filer, SQLiteDatabaseSchema schema) throws IOException {
String className = schema.getName();
String dataSourceName = PREFIX + className;
className = className.replaceAll(BindDataSourceBuilder.SUFFIX, SUFFIX);
className = PREFIX + className;
PackageElement pkg = elementUtils.getPackageOf(schema.getElement());
String packageName = pkg.isUnnamed() ? "" : pkg.getQualifiedName().toString();
AnnotationProcessorUtilis.infoOnGeneratedClasses(BindDataSource.class, packageName, className);
//@formatter:off
builder = TypeSpec.classBuilder(className).addModifiers(Modifier.PUBLIC).addModifiers(Modifier.ABSTRACT)
.addTypeVariable(TypeVariableName.get("I"))
.addTypeVariable(TypeVariableName.get("U"))
.addTypeVariable(TypeVariableName.get("R"));
//@formatter:on
// javadoc
builder.addJavadoc("\n\nSpecialized async task to make async database operation on activity\n
\n");
builder.addJavadoc("\n\nUnlike standard async task, for an instance of this class can be used many time.\n
\n");
builder.addJavadoc("\n\nWhen method execute
is invoked, an inner async task is created.\n
\n\n");
JavadocUtility.generateJavadocGeneratedBy(builder);
builder.addJavadoc("@param I input param\n");
builder.addJavadoc("@param U update param\n");
builder.addJavadoc("@param R result param\n\n");
builder.addJavadoc("@see $T\n", className(className.replaceAll(SUFFIX, BindDaoFactoryBuilder.SUFFIX)));
builder.addJavadoc("@see $T\n", className(dataSourceName));
builder.addJavadoc("@see $T\n", BindAsyncTaskType.class);
// build constructors
builder.addMethod(MethodSpec.constructorBuilder().addModifiers(Modifier.PUBLIC).addStatement("this($T.READ)", BindAsyncTaskType.class)
.addJavadoc("\nWith this constructor, a read only database connection will be used\n
\n").build());
builder.addMethod(
MethodSpec.constructorBuilder().addModifiers(Modifier.PUBLIC).addJavadoc("\nWith this constructor it is possible to specify which type of database use in async task\n
\n\n")
.addJavadoc("@param mode allows to specify if and how open a data source connection\n").addParameter(BindAsyncTaskType.class, "mode").addCode("this.mode = mode;").build());
// define fields
{
FieldSpec.Builder fieldSpec = FieldSpec.builder(BindAsyncTaskType.class, "mode", Modifier.PROTECTED);
fieldSpec.addJavadoc("Allows to specify how async task interacts with data source.\n\n");
builder.addField(fieldSpec.build());
}
{
ParameterizedTypeName parameterizedTypeName = ParameterizedTypeName.get(TypeUtility.className(AsyncTask.class), TypeUtility.typeName("I"), TypeUtility.typeName("U"),
TypeUtility.typeName("R"));
FieldSpec.Builder fieldSpec = FieldSpec.builder(parameterizedTypeName, "asyncTask", Modifier.PROTECTED);
fieldSpec.addJavadoc("Async task wrapped by this class\n\n");
builder.addField(fieldSpec.build());
}
// build methods
builder.addMethod(MethodSpec.methodBuilder("onPreExecute").addModifiers(Modifier.PUBLIC).addJavadoc("Use this method for operations on UI-thread before start execution\n").build());
builder.addMethod(MethodSpec.methodBuilder("onExecute").returns(TypeUtility.typeName("R")).addParameter(TypeUtility.typeName(dataSourceName), "dataSource")
.addJavadoc(
"Method used to encapsulate operations on datasource\n\n@param dataSource\n\tuse it to retrieve DAO\n@return\n\tresult of operation (list, bean, etc) and execute transactions.\n")
.addModifiers(Modifier.PUBLIC).addModifiers(Modifier.ABSTRACT).build());
builder.addMethod(MethodSpec.methodBuilder("onFinish").addParameter(TypeUtility.typeName("R"), "result").addModifiers(Modifier.PUBLIC).addModifiers(Modifier.ABSTRACT)
.addJavadoc("Use this method for operations on UI-thread after execution\n").build());
builder.addMethod(MethodSpec.methodBuilder("onProgressUpdate").addModifiers(Modifier.PUBLIC).addParameter(ParameterSpec.builder(ArrayTypeName.of(TypeUtility.typeName("U")), "update").addAnnotation(AnnotationSpec.builder(SuppressWarnings.class).addMember("value", "$S", "unchecked").build()).build())
.varargs().addJavadoc("Override this method to KRIPTON_DEBUG operation progress on UI-Thread\n").build());
builder.addMethod(MethodSpec.methodBuilder("onError").addParameter(Throwable.class, "exception").addModifiers(Modifier.PUBLIC)
.addJavadoc("This method is invoked when onExecute
method generate an exception.\n@param exception exception generated\n")
.addStatement("$T.error(exception.getMessage())", Logger.class).addStatement("exception.printStackTrace()").build());
// method execute1
{
MethodSpec.Builder executeBuilder = MethodSpec.methodBuilder("execute")
.addParameter(ParameterSpec.builder(ArrayTypeName.of(TypeUtility.className("I")), "params")
.addAnnotation(AnnotationSpec.builder(SuppressWarnings.class).addMember("value", "$S", "unchecked").build()).build())
.varargs(true).addModifiers(Modifier.PUBLIC).addJavadoc("Method to start operations.\n\n@param executor used executor\n@param data input\n");
executeBuilder.addStatement("executeOnExecutor($T.SERIAL_EXECUTOR, params)", android.os.AsyncTask.class);
builder.addMethod(executeBuilder.build());
}
// method execute2
{
MethodSpec.Builder executeBuilder = MethodSpec.methodBuilder("executeOnExecutor").addParameter(Executor.class, "executor")
.addParameter(ParameterSpec.builder(ArrayTypeName.of(TypeUtility.className("I")), "params")
.addAnnotation(AnnotationSpec.builder(SuppressWarnings.class).addMember("value", "$S", "unchecked").build()).build())
.varargs(true).addModifiers(Modifier.PUBLIC).addJavadoc("Method to start operations.\n\n@param executor used executor\n@param data input\n");
//@formatter:off
TypeSpec.Builder anonymous = TypeSpec.anonymousClassBuilder("")
.addSuperinterface(ParameterizedTypeName.get(TypeUtility.className(AsyncTask.class), TypeUtility.className("I"), TypeUtility.className("U"), TypeUtility.className("R")));
anonymous.addMethod(MethodSpec.methodBuilder("onPreExecute").addModifiers(Modifier.PUBLIC).addAnnotation(Override.class).addStatement("$L.this.onPreExecute()",className).build());
anonymous.addMethod(MethodSpec.methodBuilder("doInBackground").addModifiers(Modifier.PUBLIC).returns(TypeUtility.typeName("R")).addAnnotation(Override.class)
.addParameter(ParameterSpec.builder(ArrayTypeName.of(TypeUtility.className("I")), "params")
.addAnnotation(AnnotationSpec.builder(SuppressWarnings.class).addMember("value","$S", "unchecked").build())
.build()).varargs(true)
.addStatement("$L dataSource=$L.getInstance()", dataSourceName, dataSourceName)
.addStatement("R result=onExecute(dataSource)")
.addStatement("return result").build());
anonymous.addMethod(MethodSpec.methodBuilder("onProgressUpdate")
.addModifiers(Modifier.PUBLIC)
.addAnnotation(Override.class)
.addParameter(ParameterSpec.builder(ArrayTypeName.of(TypeUtility.className("U"))
, "values")
.addAnnotation(AnnotationSpec.builder(SuppressWarnings.class).addMember("value","$S", "unchecked").build())
.build())
.varargs(true)
.addStatement("$L.this.onProgressUpdate(values)", className)
.build());
anonymous.addMethod(MethodSpec.methodBuilder("onPostExecute")
.addModifiers(Modifier.PUBLIC)
.addAnnotation(Override.class)
.addParameter(ParameterSpec.builder(TypeUtility.className("R"), "result").build())
.addStatement("$L.this.onFinish(result)",className)
.build());
//@formatter:on
//@formatter:off
executeBuilder.addStatement("asyncTask=$L", anonymous.build());
executeBuilder.addStatement("asyncTask.executeOnExecutor(executor, params)", anonymous.build());
builder.addMethod(executeBuilder.build());
anonymous.addMethod(MethodSpec.methodBuilder("cancel")
.addModifiers(Modifier.PUBLIC)
.addModifiers(Modifier.FINAL)
.addParameter(Boolean.TYPE, "mayInterruptIfRunning")
.beginControlFlow("if (asyncTask!=null, args)")
.addStatement("return asyncTask.cancel(mayInterruptIfRunning)")
.endControlFlow()
.addStatement("return false").build());
}
// build SimpleAsyncTask
builder.addType(TypeSpec.classBuilder("Simple")
.addJavadoc("Simple implementation of async task. It uses read only database.\n\n")
.addJavadoc("@see $T\n", TypeUtility.typeName(className.replaceAll(SUFFIX, BindDaoFactoryBuilder.SUFFIX)))
.addJavadoc("@see $T\n", TypeUtility.typeName(dataSourceName))
.addModifiers(Modifier.PUBLIC)
.addModifiers(Modifier.STATIC)
.addModifiers(Modifier.ABSTRACT)
.addTypeVariable(TypeVariableName.get("R"))
.superclass(
ParameterizedTypeName.get(TypeUtility.className(className),
TypeUtility.className("Void"),
TypeUtility.className("Void"),
TypeUtility.className("R")
)
)
.addMethod(MethodSpec.constructorBuilder()
.addModifiers(Modifier.PUBLIC)
.addJavadoc("Create an simple async task allowing user to decide which kind of operation can be done on datasource")
.addParameter(BindAsyncTaskType.class, "mode")
.addStatement("super(mode)").build())
.addMethod(MethodSpec.constructorBuilder()
.addModifiers(Modifier.PUBLIC)
.addJavadoc("Create an simple async task for data source read only operation")
.addStatement("super($T.READ)", BindAsyncTaskType.class).build())
.build());
//@formatter:on
TypeSpec typeSpec = builder.build();
JavaWriterHelper.writeJava2File(filer, packageName, typeSpec);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy