All Downloads are FREE. Search and download functionalities are using the official Maven repository.

editor.BeanTree Maven / Gradle / Ivy

/*
 *
 *  Copyright 2010 Guidewire Software, Inc.
 *
 */
package editor;

import gw.config.CommonServices;
import gw.lang.reflect.IAttributedFeatureInfo;
import gw.lang.reflect.IMetaType;
import gw.lang.reflect.IMethodInfo;
import gw.lang.reflect.IPropertyInfo;
import gw.lang.reflect.IRelativeTypeInfo;
import gw.lang.reflect.IType;
import gw.lang.reflect.ITypeInfo;
import gw.lang.reflect.TypeInfoUtil;
import gw.lang.reflect.TypeSystem;
import gw.lang.reflect.java.JavaTypes;
import gw.util.CaseInsensitiveSet;
import gw.util.GosuStringUtil;
import gw.util.IFeatureFilter;

import javax.swing.tree.MutableTreeNode;
import javax.swing.tree.TreeNode;
import java.beans.IndexedPropertyDescriptor;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.List;
import java.util.Set;


public class BeanTree implements MutableTreeNode, Comparable
{
  private static final String ARRAY_LENGTH_PREFIX = "length = ";

  private BeanInfoNode _node;
  private TreeNode _parent;
  private List _children;
  private String _strNameConstraint;
  private boolean _bIncludeOnlySimpleDescriptors;
  private boolean _bChildrenEvaluated;
  //  private IDebugger _debugger;
  private Boolean _bIncludeStaticMembers;
  private IFeatureFilter _filter;
  private IType _whosaskin;
  private boolean _bExpansion = true;

  /**
   */
  public BeanTree( IType classBean, IType whosaskin, String strDisplayName, String strNameConstraint, Boolean bIncludeStaticMembers )
  {
    this( classBean, whosaskin, strDisplayName, strNameConstraint, bIncludeStaticMembers, null, true );
  }

  public BeanTree( IType classBean, IType whosaskin, String strDisplayName, String strNameConstraint, Boolean bIncludeStaticMembers, IFeatureFilter filter, boolean bExpansion )
  {
    _whosaskin = whosaskin;
    _node = new BeanInfoNode( classBean );
    _node.setDisplayName( strDisplayName );
    _strNameConstraint = strNameConstraint;
    _bIncludeStaticMembers = bIncludeStaticMembers;
    _filter = filter;
    _bExpansion = bExpansion;
    initializeChildren();
  }

  public BeanTree( IType classBean, IType whosaskin, String strDisplayName, boolean bIncludeOnlySimpleDescriptors, Boolean bIncludeStaticMembers )
  {
    this( classBean, whosaskin, strDisplayName, bIncludeOnlySimpleDescriptors, bIncludeStaticMembers, null, true );
  }

  /**
   */
  public BeanTree( IType classBean, IType whosaskin, String strDisplayName, boolean bIncludeOnlySimpleDescriptors, Boolean bIncludeStaticMembers, IFeatureFilter filter, boolean bExpansion )
  {
    _whosaskin = whosaskin;
    _node = new BeanInfoNode( classBean );
    _node.setDisplayName( strDisplayName );
    _bIncludeOnlySimpleDescriptors = bIncludeOnlySimpleDescriptors;
    _bIncludeStaticMembers = bIncludeStaticMembers;
    _filter = filter;
    _bExpansion = bExpansion;

    initializeChildren();
  }

  /**
   * @param classBean
   */
  public BeanTree( IType[] classBean, IType whosaskin )
  {
    this( classBean, whosaskin, false, false );
  }

  /**
   * @param classBean
   */
  public BeanTree( IType[] classBean, IType whosaskin, boolean bIncludeOnlySimpleDescriptors, Boolean bIncludeStaticMembers )
  {
    _whosaskin = whosaskin;
    _bIncludeOnlySimpleDescriptors = bIncludeOnlySimpleDescriptors;
    _bIncludeStaticMembers = bIncludeStaticMembers;
    _node = new BeanInfoNode( "Root" );
    _children = new ArrayList();

    for( int i = 0; i < classBean.length; i++ )
    {
      String strBeanClassName = classBean[i].getName();
      int iDotIndex = strBeanClassName.lastIndexOf( '.' );
      String strSymbolName = strBeanClassName.substring( iDotIndex + 1 );

      BeanTree beanTree = new BeanTree( classBean[i], whosaskin, strSymbolName, bIncludeOnlySimpleDescriptors, bIncludeStaticMembers );
      _children.add( beanTree );
      beanTree.setParent( this );
    }
  }

  BeanTree( IMethodInfo descriptor, TreeNode parent, IType whosAskin )
  {
    _whosaskin = whosAskin;
    _node = new MethodNode( descriptor );
    _parent = parent;
  }

  private BeanTree( IMethodInfo descriptor, BeanTree parent )
  {
    _whosaskin = parent._whosaskin;
    _node = new MethodNode( descriptor );
    _parent = parent;
    //## DO NOT PROPOGATE: I'm leaving this code commented out here to prevent
    //##   some clever person from making the mistake I made. Name constraints
    //##   should only be applied to the top level.
    //## _strNameConstraint = _parent._strNameConstraint;
  }

  BeanTree( ITypeInfo owner, IPropertyInfo pi, boolean arrayicize, IType whosAskin, TreeNode parent )
  {
    _whosaskin = whosAskin;
    _node = new PropertyNode( owner, pi, arrayicize, _whosaskin );
    _parent = parent;
  }

  private BeanTree( ITypeInfo owner, IPropertyInfo pi, boolean arrayicize, BeanTree parent )
  {
    _whosaskin = parent._whosaskin;
    _node = new PropertyNode( owner, pi, arrayicize, _whosaskin );
    _parent = parent;
    //## DO NOT PROPOGATE: I'm leaving this code commented out here to prevent
    //##   some clever person from making the mistake I made. Name constraints
    //##   should only be applied to the top level.
    //## _strNameConstraint = _parent._strNameConstraint;
  }

  BeanTree( BeanInfoNode node, IType whosAskin, TreeNode parent )
  {
    _whosaskin = whosAskin;
    _node = node;
    _parent = parent;
  }

  private BeanTree( BeanInfoNode node, BeanTree parent )
  {
    _whosaskin = parent._whosaskin;
    _node = node;
    _parent = parent;
    //## DO NOT PROPOGATE: I'm leaving this code commented out here to prevent
    //##   some clever person from making the mistake I made. Name constraints
    //##   should only be applied to the top level.
    //## _strNameConstraint = _parent._strNameConstraint;
  }

  //----------------------------------------------------------------------------
  public BeanInfoNode getBeanNode()
  {
    return _node;
  }

  //----------------------------------------------------------------------------
  public void setBeanNode( BeanInfoNode node )
  {
    _node = node;
  }

  /**
   *
   */
  protected void initializeChildren()
  {
    if( _children != null )
    {
      return;
    }

    _children = new ArrayList();

    IType type = _node.getType();

    if( !TypeSystem.isBeanType( type ) &&
        !(type instanceof IMetaType) &&
        type != JavaTypes.STRING() &&
        type != JavaTypes.NUMBER() &&
        !type.isArray() )
    {
      return;
    }

    try
    {
      boolean wasArray = _bExpansion && TypeSystem.isExpandable( type );
      if( wasArray )
      {
        if( type.isArray() && getBeanNode().getValue() != null )
        {
          addArrayElementNodes();
          return;
        }
        else
        {
          type = CommonServices.getTypeSystem().getExpandableComponentType( type );
        }
      }

      ITypeInfo beanInfo = null;
      try
      {
        // Note retrieving the typeinfo may throw an exception if the corresponding
        // class cannot be loaded (e.g., when the class is not in Studio's local
        // classpath, but is on the server's). We must handle this gracefully.
        beanInfo = type.getTypeInfo();
      }
      catch( Throwable t )
      {
        // ignore
      }
      if( beanInfo == null )
      {
        return;
      }

      addFeatureNodes( beanInfo, wasArray );

      Collections.sort( _children );
    }
    catch( Exception e )
    {
      editor.util.EditorUtilities.handleUncaughtException( e );
      String message = e.getMessage();
      if( GosuStringUtil.isEmpty( message ) )
      {
        editor.util.EditorUtilities.displayError( e );
      }
      else
      {
        editor.util.EditorUtilities.displayError( message );
      }
    }
  }


  private void addFeatureNodes( ITypeInfo beanInfo, boolean arrayicize )
  {
    addPropertyNodes( beanInfo, arrayicize );
    if( !excludeMethodDescriptors() && !includeOnlySimpleDescriptors() )
    {
      addMethodNodes( beanInfo );
    }
  }

  /**
   *
   */
  private void addArrayElementNodes()
  {
    String strLengthValue = getBeanNode().getValue();
    if( strLengthValue == null )
    {
      return;
    }

    int iIndex = strLengthValue.indexOf( ARRAY_LENGTH_PREFIX );
    if( iIndex < 0 )
    {
      return;
    }

    strLengthValue = strLengthValue.substring( iIndex + ARRAY_LENGTH_PREFIX.length() );
    try
    {
      int iArrayLength = Integer.parseInt( strLengthValue );
      IType atomicType = _node.getType().getComponentType();
      for( int i = 0; i < iArrayLength; i++ )
      {
        _children.add( new BeanTree( new ArrayElementNode( atomicType, i ), this ) );
      }
    }
    catch( NumberFormatException nfe )
    {
      // Eat this
    }

    evaluateProperties();
  }

  /**
   * @param beanInfo
   */
  private void addMethodNodes( ITypeInfo beanInfo )
  {
    List methods = beanInfo instanceof IRelativeTypeInfo
                                          ? ((IRelativeTypeInfo)beanInfo).getMethods( _whosaskin )
                                          : beanInfo.getMethods();
    Set signatures = new CaseInsensitiveSet();
    if( !methods.isEmpty() )
    {
      for( IMethodInfo method : methods )
      {
        if( shouldFilter( method.getDisplayName() ) )
        {
          // Filter this method -- it doesn't match the name constraint.
          continue;
        }

        if( isHidden( beanInfo, method ) )
        {
          // Don't expose hidden methods in the composer.
          continue;
        }

        if( !mutualExclusiveStaticFilter( beanInfo, method ) )
        {
          continue;
        }

        String signature = TypeInfoUtil.getMethodSignature( method );
        if( !signatures.contains( signature ) )
        {
          _children.add( new BeanTree( method, this ) );
          signatures.add( signature );
        }
      }
    }
  }

  /**
   * @param beanInfo
   * @param arrayicize
   */
  private void addPropertyNodes( ITypeInfo beanInfo, boolean arrayicize )
  {
    List properties = beanInfo instanceof IRelativeTypeInfo
                                               ? ((IRelativeTypeInfo)beanInfo).getProperties( _whosaskin )
                                               : beanInfo.getProperties();
    Set propertyNames = new CaseInsensitiveSet();
    for( IPropertyInfo property : properties )
    {
      if( shouldFilter( property.getName() ) )
      {
        // Filter this property -- it doesn't match the name constraint.
        continue;
      }

      if( property instanceof IndexedPropertyDescriptor )
      {
        IndexedPropertyDescriptor indexedProp = (IndexedPropertyDescriptor)property;
        if( indexedProp.getPropertyType() == null )
        {
          // Indexed properties must provide non-indexed access.
          continue;
        }
      }

      if( isHidden( beanInfo, property ) )
      {
        // Don't expose hidden properties in the composer.
        continue;
      }

      if( !mutualExclusiveStaticFilter( beanInfo, property ) )
      {
        continue;
      }
      if( includeOnlySimpleDescriptors() )
      {
        IType propType = property.getFeatureType();
        if( propType.isArray() || !isSimple( propType ) )
        {
          continue;
        }
      }

      String displayName = property.getDisplayName();
      if( !propertyNames.contains( displayName ) )
      {
        _children.add( new BeanTree( beanInfo, property, arrayicize && !TypeSystem.isExpandable( property.getFeatureType() ), this ) );
        propertyNames.add( displayName );
      }
    }

    evaluateProperties();
  }

  private boolean isSimple( IType cls )
  {
    return cls.isPrimitive() ||
           JavaTypes.STRING().isAssignableFrom( cls ) ||
           JavaTypes.NUMBER().isAssignableFrom( cls ) ||
           JavaTypes.BOOLEAN().isAssignableFrom( cls ) ||
           JavaTypes.DATE().isAssignableFrom( cls );
  }


  private boolean isHidden( ITypeInfo beanInfo, IAttributedFeatureInfo feature )
  {
    return feature.isHidden() ||
           !feature.isScriptable() ||
           hideDeprecated( feature ) ||
           isInternal( feature ) ||
           feature.getName().equals( "IntrinsicType" ) ||
           (getFeatureFilter() != null && !getFeatureFilter().acceptFeature( beanInfo.getOwnersType(), feature ));
  }

  private boolean mutualExclusiveStaticFilter( ITypeInfo beanInfo, IAttributedFeatureInfo descriptor )
  {
    if( includeStaticMembers() == null )
    {
      // Include both static and non-static members if null (a tri-state flag)
      return true;
    }

    if( beanInfo.getOwnersType() instanceof IMetaType ||
        includeStaticMembers() )
    {
      return descriptor.isStatic();
    }
    return !descriptor.isStatic();
  }

  private boolean isInternal( IAttributedFeatureInfo feature )
  {
    return feature instanceof IMethodInfo && feature.getDisplayName().startsWith( "@" );
  }

  private boolean shouldFilter( String strMemberName )
  {
    if( _strNameConstraint != null &&
        _strNameConstraint.length() > 0 &&
        Character.isJavaIdentifierStart( _strNameConstraint.charAt( 0 ) ) )
    {
      return !strMemberName.toLowerCase().startsWith( _strNameConstraint.toLowerCase() );
    }

    return false;
  }

  final boolean methodsEqual( Method m1, Method m2 )
  {
    if( m1 == null || m2 == null )
    {
      return false;
    }

    if( m1.getName().equals( m2.getName() ) )
    {
      Class[] params1 = m1.getParameterTypes();
      Class[] params2 = m2.getParameterTypes();
      if( params1.length == params2.length )
      {
        for( int i = 0; i < params1.length; i++ )
        {
          if( params1[i] != params2[i] )
          {
            return false;
          }
        }
        return true;
      }
    }
    return false;
  }

  @Override
  public TreeNode getChildAt( int iChildIndex )
  {
    return _children.get( iChildIndex );
  }

  @Override
  public int getChildCount()
  {
    initializeChildren();
    return _children.size();
  }

  @Override
  public TreeNode getParent()
  {
    return _parent;
  }

  public void setParent( TreeNode parent )
  {
    _parent = parent;
  }

  @Override
  public int getIndex( TreeNode node )
  {
    BeanTree beanTree = (BeanTree)node;
    return _children.indexOf( beanTree );
  }

  @Override
  public boolean getAllowsChildren()
  {
    return false;
  }

  @Override
  public boolean isLeaf()
  {
    if( _children == null )
    {
      return false;
    }
    return getChildCount() == 0;
  }

  @Override
  public Enumeration children()
  {
    initializeChildren();
    return new BeanTreeEnumeration();
  }

  @Override
  public int compareTo( BeanTree o )
  {
    if( _node == null )
    {
      return o._node == null ? 0 : -1;
    }
    else
    {
      if( o._node == null )
      {
        return 1;
      }
      return _node.compareTo( o._node );
    }
  }

  @Override
  public void insert( MutableTreeNode child, int index )
  {
    MutableTreeNode oldParent = (MutableTreeNode)child.getParent();

    if( oldParent != null )
    {
      oldParent.remove( child );
    }
    child.setParent( this );
    _children.add( index, (BeanTree)child );
  }

  @Override
  public void remove( int index )
  {
    MutableTreeNode child = (MutableTreeNode)getChildAt( index );
    _children.remove( index );
    child.setParent( null );
  }

  @Override
  public void remove( MutableTreeNode node )
  {
    //noinspection SuspiciousMethodCalls
    _children.remove( node );
    node.setParent( null );
  }

  @Override
  public void setUserObject( Object object )
  {
  }

  @Override
  public void removeFromParent()
  {
    MutableTreeNode parent = (MutableTreeNode)getParent();
    if( parent != null )
    {
      parent.remove( this );
    }
  }

  @Override
  public void setParent( MutableTreeNode newParent )
  {
    _parent = (BeanTree)newParent;
  }

  class BeanTreeEnumeration implements Enumeration
  {
    int _iCount;
    int _iChildCount = _children.size();

    public BeanTreeEnumeration()
    {
    }

    @Override
    public boolean hasMoreElements()
    {
      return (_iCount < _iChildCount);
    }

    @Override
    public Object nextElement()
    {
      if( _iCount >= _iChildCount )
      {
        return null;
      }
      else
      {
        return _children.get( _iCount++ );
      }
    }
  }

  public String makePath( boolean bFeatureLiteralCompletion )
  {
    StringBuilder path = new StringBuilder();
    TreeNode tree = this;
    while( true )
    {
      if( tree instanceof BeanTree )
      {
        BeanInfoNode node = ((BeanTree)tree).getBeanNode();
        if( node.getType() != null )
        {
          if( path.length() == 0 )
          {
            path.append( node.getPathComponent( bFeatureLiteralCompletion ) );
          }
          else
          {
            if( node.getName() != null && node.getName().length() > 0 )
            {
              path.insert( 0, '.' );
            }
            path.insert( 0, node.getName() );
          }
          tree = tree.getParent();
          continue;
        }
      }

      break;
    }

    return path.toString();
  }

  public List getChildren()
  {
    return _children;
  }

  private boolean includeOnlySimpleDescriptors()
  {
    if( _bIncludeOnlySimpleDescriptors )
    {
      return true;
    }

    if( !(_parent instanceof BeanTree) )
    {
      return false;
    }
    return ((BeanTree)_parent).includeOnlySimpleDescriptors();
  }

  private boolean excludeMethodDescriptors()
  {
    if( !(_parent instanceof BeanTree) )
    {
      return false;
    }
    return ((BeanTree)_parent).excludeMethodDescriptors();
  }

  private IFeatureFilter getFeatureFilter()
  {
    if( _filter != null )
    {
      return _filter;
    }

    if( !(_parent instanceof BeanTree) )
    {
      return null;
    }
    return ((BeanTree)_parent).getFeatureFilter();
  }


  private Boolean includeStaticMembers()
  {
    return _bIncludeStaticMembers;
  }

  private void evaluateProperties()
  {
    if( isChildrenEvaluated() )
    {
      return;
    }

    setChildrenEvaluated( true );
  }

  private String getRootPath( BeanTree tree )
  {
    if( tree.getParent().getParent() == null )
    {
      return tree.getBeanNode().getName();
    }
    if( tree.getBeanNode() instanceof ArrayElementNode )
    {
      return getRootPath( (BeanTree)tree.getParent() ) + tree.getBeanNode().getName();
    }
    else
    {
      return getRootPath( (BeanTree)tree.getParent() ) + "." + tree.getBeanNode().getName();
    }
  }

  private boolean isChildrenEvaluated()
  {
    return _bChildrenEvaluated;
  }

  private void setChildrenEvaluated( boolean bChildrenEvaluated )
  {
    _bChildrenEvaluated = bChildrenEvaluated;
  }

  private boolean hideDeprecated( IAttributedFeatureInfo descriptor )
  {
    return isHideDeprecatedMembers() &&
           descriptor.isDeprecated();
  }

  public static boolean isHideDeprecatedMembers()
  {
    return true;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy