001 /*****************************************************************************
002 * Copyright (c) PicoContainer Organization. All rights reserved. *
003 * ------------------------------------------------------------------------- *
004 * The software in this package is published under the terms of the BSD *
005 * style license a copy of which has been included with this distribution in *
006 * the LICENSE.txt file. *
007 * *
008 * Idea by Rachel Davies, Original code by Aslak Hellesoy and Paul Hammant *
009 *****************************************************************************/
010 package org.picocontainer.injectors;
011
012 import org.picocontainer.ComponentAdapter;
013 import org.picocontainer.ComponentMonitor;
014 import org.picocontainer.LifecycleStrategy;
015 import org.picocontainer.Parameter;
016 import org.picocontainer.PicoCompositionException;
017 import org.picocontainer.Characteristics;
018 import org.picocontainer.behaviors.AbstractBehaviorFactory;
019
020 import java.util.Properties;
021 import java.lang.reflect.Method;
022
023 /**
024 * A {@link org.picocontainer.InjectionFactory} for methods.
025 * The factory creates {@link MethodInjector}.
026 *
027 * @author Paul Hammant
028 */
029 @SuppressWarnings("serial")
030 public class MethodInjection extends AbstractInjectionFactory {
031
032 private final AbstractInjectionFactory delegate;
033
034 public MethodInjection(String injectionMethodName) {
035 delegate = new MethodInjectionByName(injectionMethodName);
036 }
037
038 public MethodInjection() {
039 this("inject");
040 }
041
042 public MethodInjection(Method injectionMethod) {
043 delegate = new MethodInjectionByReflectionMethod(injectionMethod);
044 }
045
046 public <T> ComponentAdapter<T> createComponentAdapter(ComponentMonitor componentMonitor, LifecycleStrategy lifecycleStrategy, Properties componentProperties, Object componentKey,
047 Class<T> componentImplementation, Parameter... parameters) throws PicoCompositionException {
048 return delegate.createComponentAdapter(componentMonitor, lifecycleStrategy, componentProperties, componentKey, componentImplementation, parameters);
049 }
050
051 public class MethodInjectionByName extends AbstractInjectionFactory {
052 private final String injectionMethodName;
053
054 public MethodInjectionByName(String injectionMethodName) {
055 this.injectionMethodName = injectionMethodName;
056 }
057
058 public <T> ComponentAdapter<T> createComponentAdapter(ComponentMonitor monitor, LifecycleStrategy lifecycleStrategy, Properties componentProperties, Object componentKey, Class<T> componentImplementation, Parameter... parameters) throws PicoCompositionException {
059 boolean useNames = AbstractBehaviorFactory.arePropertiesPresent(componentProperties, Characteristics.USE_NAMES, true);
060 return wrapLifeCycle(new MethodInjector(componentKey, componentImplementation, parameters, monitor, injectionMethodName, useNames), lifecycleStrategy);
061 }
062 }
063
064 public class MethodInjectionByReflectionMethod extends AbstractInjectionFactory {
065 private final Method injectionMethod;
066
067 public MethodInjectionByReflectionMethod(Method injectionMethod) {
068 this.injectionMethod = injectionMethod;
069 }
070
071 public <T> ComponentAdapter<T> createComponentAdapter(ComponentMonitor monitor, LifecycleStrategy lifecycleStrategy, Properties componentProperties, Object componentKey, Class<T> componentImplementation, Parameter... parameters) throws PicoCompositionException {
072 boolean useNames = AbstractBehaviorFactory.arePropertiesPresent(componentProperties, Characteristics.USE_NAMES, true);
073 if (injectionMethod.getDeclaringClass().isAssignableFrom(componentImplementation)) {
074 return wrapLifeCycle(monitor.newInjector(new MethodInjector.ByReflectionMethod(componentKey, componentImplementation, parameters, monitor, injectionMethod, useNames)), lifecycleStrategy);
075 } else {
076 throw new PicoCompositionException("method [" + injectionMethod + "] not on impl " + componentImplementation.getName());
077 }
078 }
079 }
080
081 }