/* * Copyright 2008 Niclas Hedhman. All rights Reserved. * * 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.qi4j.library.beans.support; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.util.HashMap; import org.qi4j.api.common.AppliesTo; import org.qi4j.api.common.AppliesToFilter; import org.qi4j.api.composite.Composite; import org.qi4j.api.composite.TransientBuilderFactory; import org.qi4j.api.entity.association.Association; import org.qi4j.api.entity.association.ManyAssociation; import org.qi4j.api.injection.scope.Structure; import org.qi4j.api.injection.scope.This; import org.qi4j.api.injection.scope.Uses; import org.qi4j.api.property.Property; import org.qi4j.spi.util.MethodKeyMap; @AppliesTo( { JavabeanMixin.JavabeanSupportFilter.class } ) public class JavabeanMixin implements JavabeanSupport, InvocationHandler { private HashMap<Method, Object> handlers; @Structure TransientBuilderFactory cbf; Object pojo; public JavabeanMixin( @This Composite thisComposite, @Uses Object pojo ) { this.pojo = pojo; handlers = new MethodKeyMap<Object>(); Class<? extends Composite> type = thisComposite.type(); for( Method method : type.getMethods() ) { Class<?> returnType = method.getReturnType(); if( Property.class.isAssignableFrom( returnType ) ) { JavabeanProperty prop = new JavabeanProperty( this, method ); Method pojoMethod = findMethod( pojo, prop.qualifiedName().name() ); prop.setPojoMethod( pojoMethod ); handlers.put( method, prop ); } else if( ManyAssociation.class.isAssignableFrom( returnType ) ) { JavabeanManyAssociation association = new JavabeanManyAssociation( this, method ); Method pojoMethod = findMethod( pojo, association.qualifiedName().name() ); association.setPojoMethod( pojoMethod ); handlers.put( method, association ); } else if( Association.class.isAssignableFrom( returnType ) ) { JavabeanAssociation association = new JavabeanAssociation( this, method ); association.pojoMethod = findMethod( pojo, association.qualifiedName().name() ); handlers.put( method, association ); } } } private Method findMethod( Object pojo, String name ) { String methodName = "get" + Character.toUpperCase( name.charAt( 0 ) ) + name.substring( 1 ); Method pojoMethod; try { pojoMethod = pojo.getClass().getMethod( methodName ); } catch( NoSuchMethodException e ) { methodName = "is" + Character.toUpperCase( name.charAt( 0 ) ) + name.substring( 1 ); try { pojoMethod = pojo.getClass().getMethod( methodName ); } catch( NoSuchMethodException e1 ) { throw new IllegalArgumentException( methodName + " is not present in " + pojo.getClass() ); } } return pojoMethod; } public Object getJavabean() { return pojo; } public void setJavabean( Object data ) { pojo = data; } public Object invoke( Object proxy, Method method, Object[] args ) throws Throwable { synchronized( this ) { return handlers.get( method ); } } public static class JavabeanSupportFilter implements AppliesToFilter { public boolean appliesTo( Method method, Class<?> mixin, Class<?> compositeType, Class<?> modifierClass ) { String methodName = method.getName(); Class<?> retType = method.getReturnType(); return Property.class.isAssignableFrom( retType ) || Association.class.isAssignableFrom( retType ) || ManyAssociation.class.isAssignableFrom( retType ) || "getJavabean".equals( methodName ) || "setJavabean".equals( methodName ); } } }