/** * Copyright 2011-2015 John Ericksen * * 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.androidtransfuse.experiment.generators; import com.sun.codemodel.JBlock; import com.sun.codemodel.JClassAlreadyExistsException; import org.androidtransfuse.adapter.ASTMethod; import org.androidtransfuse.adapter.ASTParameter; import org.androidtransfuse.adapter.ASTType; import org.androidtransfuse.analysis.InjectionPointFactory; import org.androidtransfuse.annotations.Factory; import org.androidtransfuse.experiment.*; import org.androidtransfuse.gen.InjectionFragmentGenerator; import org.androidtransfuse.gen.InstantiationStrategyFactory; import org.androidtransfuse.gen.variableBuilder.InjectionBindingBuilder; import org.androidtransfuse.model.InjectionNode; import org.androidtransfuse.model.InjectionNodeLogger; import org.androidtransfuse.model.MethodDescriptor; import org.androidtransfuse.util.Logger; import org.androidtransfuse.util.TransfuseRuntimeException; import org.androidtransfuse.validation.Validator; import javax.inject.Inject; /** * @author John Ericksen */ public class OnCreateInjectionGenerator implements Generation { private final ASTMethod method; private final ASTType target; private final InjectionFragmentGenerator injectionFragmentGenerator; private final InstantiationStrategyFactory instantiationStrategyFactory; private final InjectionBindingBuilder injectionBindingBuilder; private final InjectionPointFactory injectionPointFactory; private final Validator validator; private final Logger log; @Factory public interface InjectionGeneratorFactory { OnCreateInjectionGenerator build(ASTMethod method, ASTType target); } @Inject public OnCreateInjectionGenerator(/*@Assisted*/ASTMethod method, /*@Assisted*/ASTType target, InjectionFragmentGenerator injectionFragmentGenerator, InstantiationStrategyFactory instantiationStrategyFactory, InjectionBindingBuilder injectionBindingBuilder, InjectionPointFactory injectionPointFactory, Validator validator, Logger log) { this.method = method; this.injectionFragmentGenerator = injectionFragmentGenerator; this.instantiationStrategyFactory = instantiationStrategyFactory; this.target = target; this.injectionBindingBuilder = injectionBindingBuilder; this.injectionPointFactory = injectionPointFactory; this.validator = validator; this.log = log; } @Override public String getName() { return "OnCreate Injection Generator"; } @Override public void schedule(final ComponentBuilder builder, final ComponentDescriptor descriptor) { builder.add(method, GenerationPhase.INJECTION, new ComponentMethodGenerator() { @Override public void generate(MethodDescriptor methodDescriptor, JBlock block) { for (ASTParameter astParameter : methodDescriptor.getASTMethod().getParameters()) { builder.getAnalysisContext().getInjectionNodeBuilders().putType(astParameter.getASTType(), injectionBindingBuilder.buildExpression(methodDescriptor.getParameter(astParameter))); } descriptor.setRootInjectionNode(injectionPointFactory.buildInjectionNode(target, builder.getAnalysisContext())); if(validator.isInError()) { log.debug("Canceling injection generation due to error during analysis."); } else { logInjectionNodes(descriptor.getRootInjectionNode()); try { injectionFragmentGenerator.buildFragment( block, instantiationStrategyFactory.buildMethodStrategy(block, builder.getScopes()), builder.getDefinedClass(), descriptor.getRootInjectionNode(), builder.getScopes(), builder.getExpressionMap()); } catch (JClassAlreadyExistsException e) { throw new TransfuseRuntimeException("Class already exists", e); } } } }); } private void logInjectionNodes(InjectionNode injectionNode) { StringBuilder builder = new StringBuilder(); builder.append("Injection Nodes:\n"); InjectionNodeLogger logger = new InjectionNodeLogger(builder, injectionNode); while(logger.containsUnvisitedNodes()){ InjectionNode node = logger.next(); node.log(logger); logger.append("\n"); } log.debug(builder.toString()); } }