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

editor.CodeRefactorManager Maven / Gradle / Ivy

package editor;

import editor.util.TextComponentUtil;
import gw.lang.parser.IParseTree;
import gw.lang.parser.Keyword;
import gw.lang.parser.statements.IClasspathStatement;
import gw.lang.parser.statements.INamespaceStatement;
import gw.lang.parser.statements.IUsesStatement;
import gw.lang.reflect.TypeSystem;
import gw.util.GosuRefactorUtil;
import gw.util.GosuStringUtil;

import javax.swing.text.BadLocationException;
import javax.swing.text.Document;
import javax.swing.undo.CompoundEdit;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Set;

/**
 * Provides syntax-aware movements of code, including refactors, etc.
 *
 * @author cgross
 */
public class CodeRefactorManager
{
  private GosuEditor _gsEditor;

  public CodeRefactorManager( GosuEditor gsEditor )
  {
    _gsEditor = gsEditor;
  }

  public void moveSelectionUp()
  {
    moveSelection( true );
  }

  public void moveSelectionDown()
  {
    moveSelection( false );
  }

  private void moveSelection( final boolean up )
  {
    _gsEditor.waitForParser();
    moveSelectionNow( up );
  }

  private void moveSelectionNow( boolean up )
  {
    int start = _gsEditor.getEditor().getSelectionStart();
    int end = _gsEditor.getEditor().getSelectionEnd();
    String script = _gsEditor.getEditor().getText();

    //If the caret is positioned at the first point in a new line, treat it as if it is on the previous line only
    if( start != end && end > 0 && script.charAt( end - 1 ) == '\n' )
    {
      end = end - 1;
    }

    //Handle raw whitespace moves
    int rawStart = TextComponentUtil.getLineStart( script, start );
    int rawEnd = TextComponentUtil.getLineEnd( script, end );
    String line = script.substring( rawStart, rawEnd );
    if( GosuStringUtil.isWhitespace( line ) || line.trim().startsWith( "//" ) )
    {
      if( up )
      {
        if( rawStart == 0 )
        {
          return;
        }
        int lineStartBefore = TextComponentUtil.getLineStart( script, rawStart - 1 );
        GosuRefactorUtil.MoveInstruction moveInstruction = new GosuRefactorUtil.MoveInstruction( false, false, lineStartBefore );
        handleMoveInstruction( script, moveInstruction, rawStart, rawEnd, up );
        return;
      }
      else
      {
        if( rawEnd >= script.length() )
        {
          return;
        }
        int lineStartBefore = rawEnd + 1;
        GosuRefactorUtil.MoveInstruction moveInstruction = new GosuRefactorUtil.MoveInstruction( false, false, lineStartBefore );
        handleMoveInstruction( script, moveInstruction, rawStart, rawEnd, up );
        return;
      }
    }

    int peStart = TextComponentUtil.findNonWhitespacePositionAfter( script, rawStart );
    int peEnd = TextComponentUtil.findNonWhitespacePositionBefore( script, rawEnd );

    peStart = Math.min( peStart, script.length() - 1 );
    peEnd = Math.min( peEnd, script.length() - 1 );

    //Find the first statement at the selection start
    IParseTree firstStatement = GosuRefactorUtil.findFirstStatementAtLine( TextComponentUtil.getLineAtPosition( _gsEditor.getEditor(), peStart ),
                                                                           peStart,
                                                                           _gsEditor.getParser().getLocations() );

    //Find the last statement at the selection end
    IParseTree lastStatement = GosuRefactorUtil.findLastStatementAtLine( TextComponentUtil.getLineAtPosition( _gsEditor.getEditor(), peEnd ),
                                                                         peEnd,
                                                                         _gsEditor.getParser().getLocations() );

    //Find the spanning range of those two statements
    IParseTree[] boundingPair = GosuRefactorUtil.findSpanningLogicalRange( firstStatement, lastStatement );

    //If a bounding pair exists, do the move
    if( boundingPair != null )
    {
      int clipStart = Math.min( rawStart, TextComponentUtil.getLineStart( script, boundingPair[0].getOffset() ) );
      int clipEnd = Math.max( rawEnd, TextComponentUtil.getLineEnd( script, boundingPair[1].getExtent() ) );


      //if this is not a class element (that is, we are in an impl), handle moving into a white space line
      if( !GosuRefactorUtil.isClassElement( boundingPair[0] ) )
      {
        int lineStart = up ? TextComponentUtil.getWhiteSpaceOrCommentLineStartBefore( script, clipStart ) :
                        TextComponentUtil.getWhiteSpaceOrCommentLineStartAfter( script, clipEnd );
        if( lineStart != -1 )
        {
          GosuRefactorUtil.MoveInstruction moveInstruction = new GosuRefactorUtil.MoveInstruction( false, false, lineStart );
          handleMoveInstruction( script, moveInstruction, clipStart, clipEnd, up );
          return;
        }
      }

      //Not handling an into whitespace move, so do a syntax aware move
      GosuRefactorUtil.MoveInstruction moveInstruction = up ? GosuRefactorUtil.getMoveUpInstruction( boundingPair[0] ) :
                                                         GosuRefactorUtil.getMoveDownInstruction( boundingPair[1] );

      if( moveInstruction != null )
      {
        if( GosuRefactorUtil.isClassElement( boundingPair[1] ) )
        {
          IParseTree nextSibling = boundingPair[1].getNextSibling();
          if( nextSibling != null )
          {
            clipEnd = TextComponentUtil.getLineStart( script, nextSibling.getOffset() ) - 1;
            if( !up )
            {
              moveInstruction.position = TextComponentUtil.getDeepestWhiteSpaceLineStartAfter( script, moveInstruction.position );
            }
          }
        }

        handleMoveInstruction( script, moveInstruction, clipStart, clipEnd, up );
      }
    }
  }

  private void handleMoveInstruction( String script, GosuRefactorUtil.MoveInstruction moveInstruction, int startClip, int endClip, boolean up )
  {
    //Do nothing if we are moving down beyond the last position in the script
    if( endClip == script.length() - 1 && !up )
    {
      return;
    }

    if( endClip < script.length() )
    {
      endClip = endClip + 1;
    }

    String movedCodeAsString = script.substring( startClip, endClip );
    if( !movedCodeAsString.endsWith( "\n" ) )
    {
      movedCodeAsString += "\n";
    }
    int offset = moveInstruction.position;

    if( up )
    {
      offset = TextComponentUtil.getLineStart( script, offset );
    }
    else
    {
      offset = TextComponentUtil.getLineEnd( script, offset ) + 1 - movedCodeAsString.length();
    }

    StringBuilder movedCode = new StringBuilder();

    int caretPosition = _gsEditor.getEditor().getCaretPosition();
    int selectionStart = _gsEditor.getEditor().getSelectionStart();
    int selectionEnd = _gsEditor.getEditor().getSelectionEnd();

    int currentPosition = startClip;

    if( moveInstruction.indent )
    {
      String[] strings = movedCodeAsString.split( "\n" );
      for( String str : strings )
      {

        if( caretPosition > currentPosition )
        {
          caretPosition += 2;
        }
        if( selectionStart > currentPosition )
        {
          selectionStart += 2;
        }
        if( selectionEnd > currentPosition )
        {
          selectionEnd += 2;
        }

        movedCode.append( "  " );
        movedCode.append( str );
        movedCode.append( "\n" );
        currentPosition += str.length() + 1;
      }
    }
    else if( moveInstruction.outdent && movedCodeAsString.startsWith( "  " ) )
    {
      String[] strings = movedCodeAsString.split( "\n" );
      for( String str : strings )
      {
        if( str.startsWith( "  " ) )
        {
          if( caretPosition > currentPosition )
          {
            caretPosition -= 2;
          }
          if( selectionStart > currentPosition )
          {
            selectionStart -= 2;
          }
          if( selectionEnd > currentPosition )
          {
            selectionEnd -= 2;
          }
          str = str.substring( 2, str.length() );
        }

        movedCode.append( str );
        movedCode.append( "\n" );
        currentPosition += str.length() + 1;
      }
    }
    else
    {
      movedCode.append( movedCodeAsString );
    }

    CompoundEdit undoAtom = _gsEditor.getUndoManager().getUndoAtom();
    if( undoAtom != null && undoAtom.getPresentationName().equals( "Script Change" ) )
    {
      _gsEditor.getUndoManager().endUndoAtom();
    }
    _gsEditor.getUndoManager().beginUndoAtom( "moveLine" );
    try
    {

      _gsEditor.getEditor().getDocument().remove( startClip, endClip - startClip );
      _gsEditor.getEditor().getDocument().insertString( offset, movedCode.toString(), null );

      _gsEditor.getEditor().setSelectionStart( selectionStart - (startClip - offset) );
      _gsEditor.getEditor().setSelectionEnd( selectionEnd - (startClip - offset) );
      _gsEditor.getEditor().getCaret().moveDot( caretPosition - (startClip - offset) );

    }
    catch( BadLocationException e )
    {
      throw new RuntimeException( e );
    }
    finally
    {
      _gsEditor.getUndoManager().endUndoAtom();
    }
  }

  public void addToUses( String strType, boolean bTemplate, boolean bProgram )
  {
    if( isTypeUsed( strType ) )
    {
      return;
    }

    Document doc = _gsEditor.getEditor().getDocument();
    int iPos = findUsesInsertionPosition( bProgram );
    int iIndex = doc.getDefaultRootElement().getElementIndex( iPos );
    int iInsertionPt;
    String strUsesStmt = Keyword.KW_uses + " " + strType;
    if( bTemplate )
    {
      strUsesStmt = "<% " + strUsesStmt + " %>";
    }
    strUsesStmt += "\n";

    if( iPos == 0 )
    {
      iInsertionPt = iPos;
      if( !bTemplate )
      {
        strUsesStmt = strUsesStmt + "\n";
      }
    }
    else
    {
      iInsertionPt = doc.getDefaultRootElement().getElement( iIndex ).getEndOffset();
    }

    try
    {
      doc.insertString( iInsertionPt, strUsesStmt, null );
    }
    catch( BadLocationException e )
    {
      throw new RuntimeException( e );
    }
  }

  private boolean isTypeUsed( String strType )
  {
    String strRelativeName = TypePopup.getRelativeTypeName( strType );
    try
    {
      return _gsEditor.getParser().getTypeUsesMap().resolveType( strRelativeName ) != null ||
             TypeSystem.parseType( strRelativeName, _gsEditor.getParser().getTypeUsesMap().copy() ) != null;
    }
    catch( Exception e )
    {
      return false;
    }
  }

  private int findUsesInsertionPosition( boolean bProgram )
  {
    Set usesStmts = _gsEditor.getParser().getTypeUsesMap().getUsesStatements();
    int iPos = -1;
    for( Iterator iterator = usesStmts.iterator(); iterator.hasNext(); )
    {
      IUsesStatement usesStmt = (IUsesStatement)iterator.next();
      iPos = Math.max( usesStmt.getLocation().getOffset(), iPos );
    }
    if( iPos < 0 ) // No uses-stmts exists, check for a package-stmt
    {
      iPos = findPackageLocation();
      if( iPos < 0 )
      {
        if( bProgram )
        {
          // No uses-stms or package-stmt, check for a classpath-stmt
          iPos = findClasspathLocation();
          if( iPos < 0 )
          {
            // No uses-stmts, package-stmt, or classpath-stmt, insert the new uses-stmt at beginning of source
            iPos = 0;
          }
          else
          {
            // Insert after the classpath-stmt
            iPos++;
          }
        }
        else
        {
          // No uses-stmts or package-stmt, insert the new uses-stmt at beginning of source
          iPos = 0;
        }
      }
      else
      {
        // Insert after the package-stmt, that's why we increment the position
        iPos++;
      }
    }
    else if( iPos == 0 ) // Implies there exists at least one uses-stmt
    {
      // Append the new uses-statement instead of inserting it (adding 1 ensures this)
      iPos = 1;
    }
    return iPos;
  }

  private int findPackageLocation()
  {
    List locations = _gsEditor.getParser().getLocations();
    List listOut = new ArrayList();
    IParseTree.Search.getContainedParsedElementsByType( locations, INamespaceStatement.class, listOut );
    if( listOut.size() > 0 )
    {
      return listOut.get( 0 ).getLocation().getOffset();
    }
    return -1;
  }

  private int findClasspathLocation()
  {
    List locations = _gsEditor.getParser().getLocations();
    List listOut = new ArrayList();
    IParseTree.Search.getContainedParsedElementsByType( locations, IClasspathStatement.class, listOut );
    if( listOut.size() > 0 )
    {
      return listOut.get( 0 ).getLocation().getOffset();
    }
    return -1;
  }

  public void extractVariable()
  {
    ExtractVariablePopup extractVarPopup = new ExtractVariablePopup();
    extractVarPopup.showNow( _gsEditor );
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy