/*
* Copyright (C) 2011 Red Hat, Inc. and/or its affiliates.
*
* 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.jboss.errai.codegen.builder.callstack;
import java.util.Map;
import org.jboss.errai.codegen.Context;
import org.jboss.errai.codegen.RenderCacheStore;
import org.jboss.errai.codegen.Statement;
import org.jboss.errai.codegen.exception.GenerationException;
import org.jboss.errai.codegen.meta.MetaClass;
import org.jboss.errai.codegen.meta.MetaParameterizedType;
import org.jboss.errai.codegen.meta.MetaType;
/**
* {@link CallElement} to create a class reference.
*
* @author Christian Sadilek <csadilek@redhat.com>
*/
public class LoadClassReference extends AbstractCallElement {
private final MetaClass metaClass;
public LoadClassReference(final MetaClass type) {
this.metaClass = type;
}
@Override
public void handleCall(final CallWriter writer,
final Context context,
final Statement statement) {
writer.reset();
try {
nextOrReturn(writer, context, new ClassReference(metaClass));
}
catch (final GenerationException e) {
blameAndRethrow(e);
}
}
public static class ClassReference implements Statement {
private final MetaClass metaClass;
public ClassReference(final MetaClass metaClass) {
this.metaClass = metaClass;
}
@Override
public String generate(final Context context) {
return getClassReference(metaClass, context);
}
@Override
public MetaClass getType() {
return metaClass;
}
}
public static String getClassReference(final MetaType metaClass, final Context context) {
return getClassReference(metaClass, context, true);
}
private static final RenderCacheStore<MetaType, String> CLASS_LITERAL_RENDER_CACHE =
() -> "CLASS_LITERAL_RENDER_CACHE";
public static String getClassReference(final MetaType metaClass, final Context context, final boolean typeParms) {
final Map<MetaType, String> cacheStore = context.getRenderingCache(CLASS_LITERAL_RENDER_CACHE);
String result = cacheStore.get(metaClass);
if (result == null) {
result = _getClassReference(metaClass, context, typeParms);
}
return result;
}
private static String _getClassReference(final MetaType metaClass, final Context context, final boolean typeParms) {
final MetaClass erased;
if (metaClass instanceof MetaClass) {
erased = ((MetaClass) metaClass).getErased();
}
else {
return "Object";
}
String fqcn = erased.getCanonicalName();
final int idx = fqcn.lastIndexOf('.');
if (idx != -1) {
if ((context.isAutoImportActive() || "java.lang".equals(erased.getPackageName()))
&& !context.hasImport(erased)) {
context.addImport(erased);
}
if (context.hasImport(erased)) {
fqcn = fqcn.substring(idx + 1);
}
}
final StringBuilder buf = new StringBuilder(fqcn);
if (typeParms) {
buf.append(getClassReferencesForParameterizedTypes(((MetaClass) metaClass).getParameterizedType(), context));
}
return buf.toString();
}
private static final RenderCacheStore<MetaParameterizedType, String> PARMTYPE_LITERAL_RENDER_CACHE =
() -> "PARMTYPE_LITERAL_RENDER_CACHE";
private static String getClassReferencesForParameterizedTypes(final MetaParameterizedType parameterizedType,
final Context context) {
final Map<MetaParameterizedType, String> cacheStore = context.getRenderingCache(PARMTYPE_LITERAL_RENDER_CACHE);
String result = cacheStore.get(parameterizedType);
if (result == null) {
final StringBuilder buf = new StringBuilder(64);
if (parameterizedType != null && parameterizedType.getTypeParameters().length != 0) {
buf.append("<");
for (int i = 0; i < parameterizedType.getTypeParameters().length; i++) {
final MetaType typeParameter = parameterizedType.getTypeParameters()[i];
if (typeParameter instanceof MetaParameterizedType) {
final MetaParameterizedType parameterizedTypeParemeter = (MetaParameterizedType) typeParameter;
buf.append(getClassReference(parameterizedTypeParemeter.getRawType(), context));
buf.append(getClassReferencesForParameterizedTypes(parameterizedTypeParemeter, context));
}
else {
// fix to a weirdness in the GWT deferred bining API;
final String ref = getClassReference(typeParameter, context);
if ("Object".equals(ref)) {
// ignore;
return "";
}
buf.append(ref);
}
if (i + 1 < parameterizedType.getTypeParameters().length)
buf.append(", ");
}
buf.append(">");
}
result = buf.toString();
cacheStore.put(parameterizedType, result);
}
return result;
}
@Override
public String toString() {
return "[[LoadClassReference<" + metaClass.getFullyQualifiedName() + ">]" + next + "]";
}
}