View Javadoc

1   /*
2    * $Id: NewInstanceMetaMethod.java 4550 2006-12-21 22:22:40Z blackdrag $
3    *
4    * Copyright 2003 (C) James Strachan and Bob Mcwhirter. All Rights Reserved.
5    *
6    * Redistribution and use of this software and associated documentation
7    * ("Software"), with or without modification, are permitted provided that the
8    * following conditions are met:
9    *  1. Redistributions of source code must retain copyright statements and
10   * notices. Redistributions must also contain a copy of this document.
11   *  2. Redistributions in binary form must reproduce the above copyright
12   * notice, this list of conditions and the following disclaimer in the
13   * documentation and/or other materials provided with the distribution.
14   *  3. The name "groovy" must not be used to endorse or promote products
15   * derived from this Software without prior written permission of The Codehaus.
16   * For written permission, please contact info@codehaus.org.
17   *  4. Products derived from this Software may not be called "groovy" nor may
18   * "groovy" appear in their names without prior written permission of The
19   * Codehaus. "groovy" is a registered trademark of The Codehaus.
20   *  5. Due credit should be given to The Codehaus - http://groovy.codehaus.org/
21   *
22   * THIS SOFTWARE IS PROVIDED BY THE CODEHAUS AND CONTRIBUTORS ``AS IS'' AND ANY
23   * EXPRESSED OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
24   * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
25   * DISCLAIMED. IN NO EVENT SHALL THE CODEHAUS OR ITS CONTRIBUTORS BE LIABLE FOR
26   * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27   * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
28   * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29   * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30   * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31   * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
32   * DAMAGE.
33   *
34   */
35  package org.codehaus.groovy.runtime;
36  
37  import groovy.lang.MetaMethod;
38  
39  import java.lang.reflect.Modifier;
40  
41  /***
42   * A MetaMethod implementation where the underlying method is really a static
43   * helper method on some class but it appears to be an instance method on a class.
44   * 
45   * This implementation is used to add new methods to the JDK writing them as normal
46   * static methods with the first parameter being the class on which the method is added.
47   * 
48   * @author <a href="mailto:james@coredevelopers.net">James Strachan</a>
49   * @version $Revision: 4550 $
50   */
51  public class NewInstanceMetaMethod extends MetaMethod {
52  
53      private static final Class[] EMPTY_TYPE_ARRAY = {};
54      
55      private MetaMethod metaMethod;
56      private Class[] logicalParameterTypes;
57  
58      
59      public NewInstanceMetaMethod(MetaMethod metaMethod) {
60          super(metaMethod);
61          this.metaMethod = metaMethod;
62          init();
63      }
64      
65      public NewInstanceMetaMethod(String name, Class declaringClass, Class[] parameterTypes, Class returnType, int modifiers) {
66          super(name, declaringClass, parameterTypes, returnType, modifiers);
67          this.metaMethod = new MetaMethod(name, declaringClass, parameterTypes,returnType, modifiers);
68          init();
69      }
70      
71      private void init() {
72          Class[] realParameterTypes = metaMethod.getParameterTypes();
73          int size = realParameterTypes!=null ? realParameterTypes.length : 0;
74          if (size <= 1) {
75              logicalParameterTypes = EMPTY_TYPE_ARRAY;
76          } else {
77              logicalParameterTypes = new Class[--size];
78              System.arraycopy(realParameterTypes, 1, logicalParameterTypes, 0, size);
79          }
80      }
81      
82      public Class getDeclaringClass() {
83          return getBytecodeParameterTypes()[0];
84      }
85  
86      public boolean isStatic() {
87          return false;
88      }
89  
90      public int getModifiers() {
91          // lets clear the static bit
92          return super.getModifiers() ^ Modifier.STATIC;
93      }
94  
95      public Class[] getParameterTypes() {
96          return logicalParameterTypes;
97      }
98  
99      public Class[] getBytecodeParameterTypes() {
100         return super.getParameterTypes();
101     }
102 
103     public Object invoke(Object object, Object[] arguments)  {
104         // we need to cheat using the type
105         int size = arguments.length;
106         Object[] newArguments = new Object[size + 1];
107         newArguments[0] = object;
108         System.arraycopy(arguments, 0, newArguments, 1, size);
109         return metaMethod.invoke(null, newArguments);
110     }
111     
112     public Class getOwnerClass() {
113         return getBytecodeParameterTypes()[0];
114     }
115 }