/***** BEGIN LICENSE BLOCK ***** * Version: CPL 1.0/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Common Public * License Version 1.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.eclipse.org/legal/cpl-v10.html * * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. * * Copyright (C) 2006 Lukas Felber <lfelber@hsr.ch> * Copyright (C) 2006 Mirko Stocker <me@misto.ch> * Copyright (C) 2006 Thomas Corbat <tcorbat@hsr.ch> * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the CPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the CPL, the GPL or the LGPL. ***** END LICENSE BLOCK *****/ package org.rubypeople.rdt.refactoring.core.movemethod; import org.jruby.ast.ArrayNode; import org.jruby.ast.Node; import org.rubypeople.rdt.refactoring.core.NodeFactory; import org.rubypeople.rdt.refactoring.core.NodeProvider; import org.rubypeople.rdt.refactoring.editprovider.ReplaceEditProvider; import org.rubypeople.rdt.refactoring.nodewrapper.ArgsNodeWrapper; import org.rubypeople.rdt.refactoring.nodewrapper.MethodNodeWrapper; public class DelegateMethodEditProvider extends ReplaceEditProvider { private MoveMethodConfig config; private Node scopePos; private MethodNodeWrapper oldMethod; public DelegateMethodEditProvider(MoveMethodConfig config) { super(false); this.config = config; oldMethod = config.getMethodNode(); scopePos = NodeProvider.unwrap(oldMethod.getBodyNode()); } @Override protected int getOffsetLength(String document) { // HACK For some nodes we're getting a position that is one too far off the end, making us gobble up a newline! So we hack a fix here int length = getOffsetLength(); int offset = getOffset(document); String sub = document.substring(offset, offset + length); if (sub.endsWith("\n")) return length - 1; return length; } @Override protected int getOffsetLength() { return getExtendedPosition(scopePos).getEndOffset() - getExtendedPosition(scopePos).getStartOffset(); } @Override protected Node getEditNode(int offset, String document) { return NodeFactory.createNewLineNode(getMethodCallNode()); } private Node getMethodCallNode() { Node receiverNode; if (oldMethod.isClassMethod()) { receiverNode = NodeFactory.createConstNode(config.getDestinationClassNode().getName()); } else { receiverNode = NodeFactory.createInstVarNode(config.getFieldInSourceClassOfTypeDestinationClass()); } ArgsNodeWrapper argsNode = oldMethod.getArgsNode(); ArrayNode arrayNode = NodeFactory.createArrayNode(); for (String argName : argsNode.getArgsList()) { arrayNode.add(NodeFactory.createLocalVarNode(argName)); } if (config.doesNewMethodNeedsReferenceToSourceClass()) { arrayNode.add(NodeFactory.createSelfNode()); } return NodeFactory.createCallNode(receiverNode, config.getMovedMethodName(), (arrayNode.size() == 0) ? null : arrayNode); } @Override protected int getOffset(String document) { return getExtendedPosition(scopePos).getStartOffset(); } }