View Javadoc
1   /**
2    * Copyright 2005-2014 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.krad.uif.component;
17  
18  import org.apache.commons.lang.StringUtils;
19  import org.kuali.rice.core.api.exception.RiceRuntimeException;
20  import org.kuali.rice.krad.datadictionary.Copyable;
21  import org.kuali.rice.krad.datadictionary.parse.BeanTag;
22  import org.kuali.rice.krad.datadictionary.parse.BeanTagAttribute;
23  import org.kuali.rice.krad.uif.util.CopyUtils;
24  import org.springframework.util.MethodInvoker;
25  import org.springframework.util.ReflectionUtils;
26  
27  import java.io.Serializable;
28  import java.lang.reflect.Method;
29  
30  /**
31   * Extends <code>MethodInvoker</code> to add properties for specifying
32   * a method for invocation within the UIF
33   *
34   * @author Kuali Rice Team (rice.collab@kuali.org)
35   */
36  @BeanTag(name = "methodInvokerConfig-bean", parent = "Uif-MethodInvokerConfig")
37  public class MethodInvokerConfig extends MethodInvoker implements Serializable, Copyable {
38      private static final long serialVersionUID = 6626790175367500081L;
39  
40      private String staticMethod;
41      private Class[] argumentTypes;
42  
43      /**
44       * {@inheritDoc}
45       */
46      @Override
47      public void prepare() throws ClassNotFoundException, NoSuchMethodException {
48          if ((getTargetObject() == null) && (getTargetClass() != null)) {
49              try {
50                  setTargetObject(getTargetClass().newInstance());
51              } catch (Exception e) {
52                  throw new RiceRuntimeException("Unable to create new intance of target class", e);
53              }
54          }
55  
56          super.prepare();
57      }
58  
59      /**
60       * Set a fully qualified static method name to invoke,
61       * e.g. "example.MyExampleClass.myExampleMethod".
62       * Convenient alternative to specifying targetClass and targetMethod.
63       *
64       * @return static method to invoke
65       */
66      @BeanTagAttribute(name="staticMethod")
67      public String getStaticMethod() {
68          return staticMethod;
69      }
70  
71      /**
72       * Override to catch a set staticMethod since super does
73       * not contain a getter
74       *
75       * @param staticMethod static method to invoke
76       */
77      @Override
78      public void setStaticMethod(String staticMethod) {
79          super.setStaticMethod(staticMethod);
80          this.staticMethod = staticMethod;
81      }
82  
83      /**
84       * Declared argument types for the method to be invoked, if not set the types will
85       * be retrieved based on the target class and target name
86       *
87       * @return method argument types
88       */
89      @BeanTagAttribute(name="argumentTypes",type= BeanTagAttribute.AttributeType.LISTBEAN)
90      public Class[] getArgumentTypes() {
91          if (argumentTypes == null) {
92              return getMethodArgumentTypes();
93          }
94  
95          return argumentTypes;
96      }
97  
98      /**
99       * Setter for the method argument types that should be invoked
100      *
101      * @param argumentTypes
102      */
103     public void setArgumentTypes(Class[] argumentTypes) {
104         this.argumentTypes = argumentTypes;
105     }
106 
107     /**
108      * Finds the method on the target class that matches the target name and
109      * returns the declared parameter types
110      *
111      * @return method parameter types
112      */
113     protected Class[] getMethodArgumentTypes() {
114         if (StringUtils.isNotBlank(staticMethod)) {
115             int lastDotIndex = this.staticMethod.lastIndexOf('.');
116             if (lastDotIndex == -1 || lastDotIndex == this.staticMethod.length()) {
117                 throw new IllegalArgumentException("staticMethod must be a fully qualified class plus method name: " +
118                         "e.g. 'example.MyExampleClass.myExampleMethod'");
119             }
120             String className = this.staticMethod.substring(0, lastDotIndex);
121             String methodName = this.staticMethod.substring(lastDotIndex + 1);
122             try {
123                 setTargetClass(resolveClassName(className));
124             } catch (ClassNotFoundException e) {
125                 throw new RuntimeException("Unable to get class for name: " + className);
126             }
127             setTargetMethod(methodName);
128         }
129 
130         Method matchingCandidate = findMatchingMethod();
131         if (matchingCandidate != null) {
132             return matchingCandidate.getParameterTypes();
133         }
134 
135         Method[] candidates = ReflectionUtils.getAllDeclaredMethods(getTargetClass());
136         for (Method candidate : candidates) {
137             if (candidate.getName().equals(getTargetMethod())) {
138                 return candidate.getParameterTypes();
139             }
140         }
141 
142         return null;
143     }
144 
145     /**
146      * @see Copyable#clone()
147      */
148     @Override
149     public MethodInvokerConfig clone() throws CloneNotSupportedException {
150         return (MethodInvokerConfig) super.clone();
151     }
152 
153     /**
154      * @see Copyable#copy()
155      * @see CopyUtils#copy(Copyable)
156      */
157     public final <T> T copy() {
158         return CopyUtils.copy(this);
159     }
160 
161     /**
162      * {@inheritDoc}
163      */
164     @Override
165     public Copyable unwrap() {
166         return this;
167     }
168 
169     /**
170      * Modification is not controlled at this level.
171      * 
172      * {@inheritDoc}
173      */
174     @Override
175     public void preventModification() {}
176 
177 }