com.codename1.maven.StubGenerator Maven / Gradle / Ivy
The newest version!
/*
* Copyright (c) 2012, Codename One and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Codename One designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Codename One through http://www.codenameone.com/ if you
* need additional information or have any questions.
*/
package com.codename1.maven;
import org.apache.maven.plugin.logging.Log;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import static com.codename1.maven.PathUtil.path;
/**
* Simple tool to generate platform implementation stubs matching a
* Native interface instance
*
* This is used by the {@link GenerateNativeInterfaces} mojo.
*/
public class StubGenerator {
private Class nativeInterface;
private StubGenerator() {}
private File androidFile;
private File javaseFile;
private File csFile;
private File iosHFile;
private File iosMFile;
private File jsFile;
private Log log;
public static StubGenerator create(Log log, Class nativeInterface) {
StubGenerator instance = new StubGenerator();
instance.log = log;
instance.nativeInterface = nativeInterface;
return instance;
}
private boolean isSubinterfaceOfNativeInterface() {
for(Class current : nativeInterface.getInterfaces()) {
if(current.getName().equals("com.codename1.system.NativeInterface")) {
return true;
}
}
return false;
}
/**
* Checks that the native interface is valid and if not returns the error as a string
* returns null for a valid interface.
*/
public String verify() {
if(!nativeInterface.isInterface()) {
return "Not an interface! Native interfaces must be interfaces.";
}
if(!isSubinterfaceOfNativeInterface()) {
return "The interface MUST implement NativeInterface!";
}
if((nativeInterface.getModifiers() & Modifier.PUBLIC) != Modifier.PUBLIC) {
return "The interface must be a public interface and not an inner class";
}
if(nativeInterface.getEnclosingClass() != null) {
return "The interface must be a public interface and not an inner class";
}
Method[] mtds = nativeInterface.getMethods();
int offset = 0;
for(Method m : mtds) {
if(m.getExceptionTypes().length > 0) {
return "Exceptions aren't supported when communicating with native interfaces, in the method " + m.getName();
}
if(m.getName().equalsIgnoreCase("init")) {
return "init() is a reserved method in iOS (a constructor of sort) naming a method init will not work properly.";
}
if(!isValidType(m.getReturnType())) {
return "Unsupported return type " + m.getReturnType().getSimpleName() + " in the method " + m.getName();
}
if(!checkDuplicateName(mtds, offset)) {
return "A method with the same name exists for the method " + m.getName() +
", notice that duplicate names (even with different case) aren't supported!";
}
offset++;
for(Class arg : m.getParameterTypes()) {
if(!isValidType(arg)) {
return "Unsupported argument type " + arg.getSimpleName() + " in the method " + m.getName();
}
}
}
return null;
}
private boolean checkDuplicateName(Method[] mtds, int offset) {
String name = mtds[offset].getName();
for(int iter = offset + 1 ; iter < mtds.length ; iter++) {
if(mtds[iter].getName().equalsIgnoreCase(name)) {
return false;
}
}
return true;
}
private void initFileNames(File destination) {
String className = nativeInterface.getName().replace('.', File.separatorChar) + "Impl.java";
androidFile = new File(path(destination.getAbsolutePath(), "android", "src", "main", "java", className));
androidFile.getParentFile().mkdirs();
javaseFile = new File(path(destination.getAbsolutePath(), "javase", "src", "main", "java", className));
javaseFile.getParentFile().mkdirs();
csFile = new File(path(destination.getAbsolutePath(), "win", "src", "main", "csharp", nativeInterface.getName().replace('.', File.separatorChar) + "Impl.cs"));
csFile.getParentFile().mkdirs();
String iosFilename = nativeInterface.getName().replace('.', '_') + "Impl.";
iosHFile = new File(path(destination.getAbsolutePath(), "ios", "src", "main", "objectivec", iosFilename+"h"));
iosMFile = new File(path(destination.getAbsolutePath(), "ios", "src", "main", "objectivec", iosFilename+"m"));
iosMFile.getParentFile().mkdirs();
jsFile = new File(path(destination.getAbsolutePath(), "javascript", "src", "main", "javascript", nativeInterface.getName().replace('.', '_') + ".js"));
jsFile.getParentFile().mkdirs();
}
/**
* Returns true if native files might need overwriting
*/
public boolean isFilesExist(File destination) {
initFileNames(destination);
return androidFile.exists() || iosHFile.exists() || iosMFile.exists() ||
csFile.exists() || javaseFile.exists() || jsFile.exists();
}
/**
* Generates the code to the given base directory
*/
public void generateCode(File destination, boolean overwrite) throws IOException {
initFileNames(destination);
if(overwrite || !androidFile.exists()) {
log.info("Writing "+androidFile);
generateJavaFile(androidFile, "android.view.View", false);
} else {
log.debug(androidFile+" already exists. Skipping");
}
if(overwrite || !javaseFile.exists()) {
log.info("Writing "+javaseFile);
generateJavaFile(javaseFile, "com.codename1.ui.PeerComponent", true);
} else {
log.debug(javaseFile+" already exists. Skipping");
}
if(overwrite || !csFile.exists()) {
log.info("Writing "+csFile);
generateCSFile(csFile, "FrameworkElement");
} else {
log.debug(csFile+" already exists. Skipping");
}
if(overwrite || !(iosHFile.exists() || iosMFile.exists())) {
log.info("Writing "+iosHFile);
log.info("Writing "+iosMFile);
generateIOSFiles();
} else {
log.debug(iosHFile+" already exists. Skipping");
}
if(overwrite || !(jsFile.exists())) {
log.info("Writing "+jsFile);
generateJavaScriptFile();
} else {
log.debug(jsFile+" already exists. Skipping");
}
}
private void generateIOSFiles() throws IOException {
String h = "#import \n\n"
+ "@interface " + nativeInterface.getName().replace('.', '_') + "Impl : NSObject {\n"
+ "}\n\n";
Method[] mtds = nativeInterface.getMethods();
for(Method m : mtds) {
h += "-(" + javaTypeToObjectiveCType(m.getReturnType()) +")" + m.getName();
Class[] params = m.getParameterTypes();
if(params == null || params.length == 0) {
h += ";\n";
continue;
}
h += ":(" + javaTypeToObjectiveCType(params[0]) + ")param";
if(params.length == 1) {
h += ";\n";
continue;
}
for(int iter = 1 ; iter < params.length ; iter++) {
h += " param" + iter + ":(" + javaTypeToObjectiveCType(params[iter]) + ")param" + iter;
}
h += ";\n";
}
h += "@end\n";
String m = "#import \"" + nativeInterface.getName().replace('.', '_') + "Impl.h\"\n\n"
+ "@implementation " + nativeInterface.getName().replace('.', '_') + "Impl\n\n";
for(Method mtd : mtds) {
Class returnType = mtd.getReturnType();
m += "-(" + javaTypeToObjectiveCType(returnType) +")" + mtd.getName();
Class[] params = mtd.getParameterTypes();
if(params == null || params.length == 0) {
m += "{\n";
} else {
m += ":(" + javaTypeToObjectiveCType(params[0]) + ")param";
if(params.length == 1) {
m += "{\n";
} else {
for(int iter = 1 ; iter < params.length ; iter++) {
m += " param" + iter + ":(" + javaTypeToObjectiveCType(params[iter]) + ")param" + iter;
}
m += "{\n";
}
}
if(returnType != Void.TYPE && returnType != Void.class) {
if(returnType == String.class || returnType.getName().equals("com.codename1.ui.PeerComponent") ||
returnType.isArray()) {
m += " return nil;\n";
} else {
if(returnType == Boolean.class || returnType == Boolean.TYPE) {
m += " return NO;\n";
} else {
m += " return 0;\n";
}
}
}
m+= "}\n\n";
}
m += "@end\n";
iosHFile.getParentFile().mkdirs();
FileOutputStream fo = new FileOutputStream(iosHFile);
fo.write(h.getBytes());
fo.close();
fo = new FileOutputStream(iosMFile);
fo.write(m.getBytes());
fo.close();
}
private String javaTypeToObjectiveCType(Class t) {
if(t == String.class) {
return "NSString*";
}
if(t.isArray()) {
return "NSData*";
}
if(t == Integer.class || t == Integer.TYPE) {
return "int";
}
if(Long.class == t || Long.TYPE == t) {
return "long long";
}
if(Byte.class == t || Byte.TYPE == t) {
return "char";
}
if(Short.class == t || Short.TYPE == t) {
return "short";
}
if(Character.class == t || Character.TYPE == t) {
return "int";
}
if(Boolean.class == t || Boolean.TYPE == t) {
return "BOOL";
}
if(Float.class == t || Float.TYPE == t) {
return "float";
}
if(Double.class == t || Double.TYPE == t) {
return "double";
}
if(Void.class == t || Void.TYPE == t) {
return "void";
}
return "void*";
}
private String javaTypeToCSharpType(Class t) {
if(t.getName().equals("com.codename1.ui.PeerComponent")) {
return "object";
}
if(t == String.class) {
return "string ";
}
if(t == int.class || t == Integer.class || t == Integer.TYPE) {
if(t.isArray()) {
return "int[]";
}
return "int";
}
if(t == long.class || Long.class == t || Long.TYPE == t) {
if(t.isArray()) {
return "long[]";
}
return "long";
}
if(t == byte.class || Byte.class == t || Byte.TYPE == t) {
if(t.isArray()) {
return "byte[]";
}
return "byte";
}
if(t == short.class || Short.class == t || Short.TYPE == t) {
if(t.isArray()) {
return "short[]";
}
return "short";
}
if(t == char.class || Character.class == t || Character.TYPE == t) {
if(t.isArray()) {
return "char[]";
}
return "char";
}
if(t == boolean.class || Boolean.class == t || Boolean.TYPE == t) {
if(t.isArray()) {
return "bool[]";
}
return "bool";
}
if(t == float.class || Float.class == t || Float.TYPE == t) {
if(t.isArray()) {
return "float[]";
}
return "float";
}
if(t == double.class || Double.class == t || Double.TYPE == t) {
if(t.isArray()) {
return "double[]";
}
return "double";
}
if(Void.class == t || Void.TYPE == t) {
return "void";
}
return t.getSimpleName();
}
private void generateJavaFile(File dest, String peerComponentType, boolean impl) throws IOException {
String t = "package " + nativeInterface.getPackage().getName() + ";\n\n"
+ "public class " + nativeInterface.getSimpleName() + "Impl ";
if(impl) {
t += "implements " + nativeInterface.getName() + "{\n";
} else {
t += "{\n";
}
Method[] mtds = nativeInterface.getMethods();
for(Method m : mtds) {
t += " public ";
Class returnType = m.getReturnType();
if(returnType.getName().equals("com.codename1.ui.PeerComponent")) {
t += peerComponentType;
} else {
t += returnType.getSimpleName();
}
t += " " + m.getName() + "(";
int offset = 0;
for(Class arg : m.getParameterTypes()) {
String s = arg.getSimpleName();
if(arg.getName().equals("com.codename1.ui.PeerComponent")) {
s = peerComponentType;
}
if(offset == 0) {
t += s + " param";
} else {
t += ", " + s + " param" + offset;
}
offset++;
}
t += ") {\n";
if(returnType != Void.TYPE && returnType != Void.class) {
if(returnType == String.class || returnType.getName().equals("com.codename1.ui.PeerComponent") ||
returnType.isArray()) {
t += " return null;\n";
} else {
if(returnType == Boolean.class || returnType == Boolean.TYPE) {
t += " return false;\n";
} else {
if(returnType == Character.class || returnType == Character.TYPE) {
t += " return (char)0;\n";
} else {
if(returnType == Byte.class || returnType == Byte.TYPE) {
t += " return (byte)0;\n";
} else {
if(returnType == Short.class || returnType == Short.TYPE) {
t += " return (short)0;\n";
} else {
t += " return 0;\n";
}
}
}
}
}
}
t += " }\n\n";
}
t += "}\n";
dest.getParentFile().mkdirs();
FileOutputStream fo = new FileOutputStream(dest);
fo.write(t.getBytes());
fo.close();
}
private void generateCSFile(File dest, String peerComponentType) throws IOException {
String t = "namespace " + nativeInterface.getPackage().getName() + "{\r\n\r\n" +
"\r\n" +
"public class " + nativeInterface.getSimpleName() + "Impl : I"+ nativeInterface.getSimpleName() + "Impl {\r\n";
Method[] mtds = nativeInterface.getMethods();
for(Method m : mtds) {
t += " public ";
Class returnType = m.getReturnType();
t += javaTypeToCSharpType(returnType);
t += " " + m.getName() + "(";
int offset = 0;
for(Class arg : m.getParameterTypes()) {
String s = arg.getSimpleName();
if(arg.getName().equals("com.codename1.ui.PeerComponent")) {
s = "object";
} else {
if(arg == boolean.class || arg == Boolean.class || arg == Boolean.TYPE) {
s = "bool";
}
}
if(offset == 0) {
t += s + " param";
} else {
t += ", " + s + " param" + offset;
}
offset++;
}
t += ") {\n";
if(returnType != Void.TYPE && returnType != Void.class) {
if(returnType == String.class || returnType.getName().equals("com.codename1.ui.PeerComponent") ||
returnType.isArray()) {
t += " return null;\n";
} else {
if(returnType == Boolean.class || returnType == Boolean.TYPE) {
t += " return false;\n";
} else {
if(returnType == Character.class || returnType == Character.TYPE) {
t += " return (char)0;\n";
} else {
if(returnType == Byte.class || returnType == Byte.TYPE) {
t += " return (byte)0;\n";
} else {
if(returnType == Short.class || returnType == Short.TYPE) {
t += " return (short)0;\n";
} else {
t += " return 0;\n";
}
}
}
}
}
}
t += " }\n\n";
}
t += "}\r\n}\r\n";
dest.getParentFile().mkdirs();
FileOutputStream fo = new FileOutputStream(dest);
fo.write(t.getBytes());
fo.close();
}
private boolean isValidType(Class cls) {
if(cls.isPrimitive()) {
return true;
}
if(cls.isArray()) {
return cls.getComponentType().isPrimitive();
}
if(cls == String.class) {
return true;
}
if(cls.getName().equals("com.codename1.ui.PeerComponent")) {
return true;
}
return false;
}
private String typeToXMLVMJavaName(Class type) {
if(type.isArray()) {
return getSimpleNameWithJavaLang(type.getComponentType()).replace('.', '_') + "_1ARRAY";
}
return getSimpleNameWithJavaLang(type).replace('.', '_');
}
private String getSimpleNameWithJavaLang(Class c) {
if(c.isPrimitive()) {
return c.getSimpleName();
}
if(c.isArray()) {
return getSimpleNameWithJavaLang(c.getComponentType()) + "[]";
}
if(c.getClass().getName().startsWith("java.lang.")) {
return c.getName();
}
return c.getSimpleName();
}
private void generateJavaScriptFile() throws IOException {
String t = "(function(exports){\n\n" +
"var o = {};\n\n";
Method[] mtds = nativeInterface.getMethods();
for(Method m : mtds) {
t += " o.";
t += m.getName();
t += "_";
Class[] params = m.getParameterTypes();
for(Class currentParam : params) {
if(currentParam.getName().equals("com.codename1.ui.PeerComponent")) {
t += "_com_codename1_ui_PeerComponent";
} else {
t += "_";
t += typeToXMLVMJavaName(currentParam);
}
}
t += " = function(";
if(params.length > 0) {
t += "param1";
for(int iter = 2 ; iter < params.length + 1 ; iter++) {
t += ", param" + iter;
}
t += ", callback) {\n";
} else {
t += "callback) {\n";
}
if(m.getName().equals("isSupported")) {
t += " callback.complete(false);\n";
} else {
t += " callback.error(new Error(\"Not implemented yet\"));\n";
}
t += " };\n\n";
}
t += "exports.";
t += nativeInterface.getName().replace('.', '_');
t += "= o;\n\n" +
"})(cn1_get_native_interfaces());\n";
FileOutputStream fo = new FileOutputStream(jsFile);
fo.write(t.getBytes());
fo.close();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy