/*=============================================================================#
# Copyright (c) 2008-2016 Stephan Wahlbrink (WalWare.de) and others.
# All rights reserved. This program and the accompanying materials
# are made available under the terms of the Eclipse Public License v1.0
# which accompanies this distribution, and is available at
# http://www.eclipse.org/legal/epl-v10.html
#
# Contributors:
# Stephan Wahlbrink - initial API and implementation
#=============================================================================*/
package de.walware.statet.r.internal.ui.editors;
import java.util.List;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.jface.text.IRegion;
import de.walware.jcommons.collections.ImCollections;
import de.walware.jcommons.collections.ImIdentityList;
import de.walware.ecommons.ltk.LTKUtil;
import de.walware.ecommons.ltk.core.model.ISourceStructElement;
import de.walware.ecommons.ltk.ui.sourceediting.assist.AssistInvocationContext;
import de.walware.ecommons.ltk.ui.sourceediting.assist.AssistProposalCollector;
import de.walware.ecommons.ltk.ui.sourceediting.assist.IQuickAssistComputer;
import de.walware.statet.r.core.model.IRCompositeSourceElement;
import de.walware.statet.r.core.model.IRLangSourceElement;
import de.walware.statet.r.core.model.IRSourceUnit;
import de.walware.statet.r.core.model.IRWorkspaceSourceUnit;
import de.walware.statet.r.core.model.RElementAccess;
import de.walware.statet.r.core.rsource.ast.NodeType;
import de.walware.statet.r.core.rsource.ast.RAstNode;
import de.walware.statet.r.internal.ui.correction.RLinkedNamesAssistProposal;
import de.walware.statet.r.internal.ui.correction.RenameInRegionAssistProposal;
import de.walware.statet.r.internal.ui.correction.RenameInWorkspaceAssistProposal;
public class RQuickRefactoringComputer implements IQuickAssistComputer {
public RQuickRefactoringComputer() {
}
@Override
public IStatus computeAssistProposals(final AssistInvocationContext context,
final AssistProposalCollector proposals, final IProgressMonitor monitor) {
if (!(context.getAstSelection().getCovering() instanceof RAstNode)) {
return Status.OK_STATUS;
}
final RAstNode node = (RAstNode) context.getAstSelection().getCovering();
if (node.getNodeType() == NodeType.SYMBOL || node.getNodeType() == NodeType.STRING_CONST) {
RAstNode candidate = node;
SEARCH_ACCESS : while (candidate != null) {
final List<Object> attachments= candidate.getAttachments();
for (final Object attachment : attachments) {
if (attachment instanceof RElementAccess) {
RElementAccess access= (RElementAccess) attachment;
SUB: while (access != null) {
if (access.getSegmentName() == null) {
break SUB;
}
if (access.getNameNode() == node) {
addAccessAssistProposals(context, access, proposals);
break SEARCH_ACCESS;
}
access = access.getNextSegment();
}
}
}
candidate = candidate.getRParent();
}
}
else if (context.getLength() > 0 && context.getSourceUnit() instanceof IRSourceUnit) {
proposals.add(new RenameInRegionAssistProposal(context));
}
return Status.OK_STATUS;
}
protected void addAccessAssistProposals(final AssistInvocationContext context,
final RElementAccess access,
final AssistProposalCollector proposals) {
final ImIdentityList<? extends RElementAccess> allAccess= ImCollections.toIdentityList(
access.getAllInUnit(false) );
proposals.add(new RLinkedNamesAssistProposal(RLinkedNamesAssistProposal.IN_FILE, context, access));
if (allAccess.size() > 2) {
IRegion chunk= null;
{ final ISourceStructElement sourceElement= context.getModelInfo().getSourceElement();
if (sourceElement instanceof IRCompositeSourceElement) {
final List<? extends IRLangSourceElement> elements= ((IRCompositeSourceElement) sourceElement).getCompositeElements();
final IRLangSourceElement element= LTKUtil.getCoveringSourceElement(elements, access.getNameNode().getOffset());
if (element != null) {
chunk= element.getSourceRange();
}
}
}
final int current= allAccess.indexOf(access);
if (current >= 0) {
if (current > 0 && current < allAccess.size() - 1) {
proposals.add(new RLinkedNamesAssistProposal(RLinkedNamesAssistProposal.IN_FILE_PRECEDING, context, access));
proposals.add(new RLinkedNamesAssistProposal(RLinkedNamesAssistProposal.IN_FILE_FOLLOWING, context, access));
}
if (chunk != null) {
int chunkBegin= 0;
for (final int offset= chunk.getOffset();
chunkBegin < current; chunkBegin++) {
if (offset <= allAccess.get(chunkBegin).getNameNode().getOffset()) {
break;
}
}
int chunkEnd= current + 1;
for (final int offset= chunk.getOffset() + chunk.getLength();
chunkEnd < allAccess.size(); chunkEnd++) {
if (offset <= allAccess.get(chunkEnd).getNameNode().getOffset()) {
break;
}
}
if (chunkEnd - chunkBegin > 1) {
proposals.add(new RLinkedNamesAssistProposal(RLinkedNamesAssistProposal.IN_CHUNK, context, access, chunk));
}
}
}
}
if (context.getSourceUnit() instanceof IRWorkspaceSourceUnit) {
proposals.add(new RenameInWorkspaceAssistProposal(context, access.getNameNode()));
}
}
}