| /******************************************************************************* |
| * Copyright (c) 2013 Red Hat Inc. |
| * 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: |
| * Red Hat Inc. - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.linuxtools.internal.systemtap.ui.ide.handlers; |
| |
| import org.eclipse.core.commands.AbstractHandler; |
| import org.eclipse.core.commands.ExecutionEvent; |
| import org.eclipse.jface.dialogs.MessageDialog; |
| import org.eclipse.jface.text.BadLocationException; |
| import org.eclipse.jface.text.IDocument; |
| import org.eclipse.jface.text.IRegion; |
| import org.eclipse.jface.text.ITextOperationTarget; |
| import org.eclipse.jface.text.ITextSelection; |
| import org.eclipse.jface.text.ITypedRegion; |
| import org.eclipse.jface.text.Region; |
| import org.eclipse.jface.text.TextUtilities; |
| import org.eclipse.jface.viewers.ISelection; |
| import org.eclipse.linuxtools.internal.systemtap.ui.ide.Localization; |
| import org.eclipse.linuxtools.internal.systemtap.ui.ide.editors.stp.STPPartitionScanner; |
| import org.eclipse.linuxtools.systemtap.graphing.ui.widgets.ExceptionErrorDialog; |
| import org.eclipse.swt.custom.BusyIndicator; |
| import org.eclipse.swt.widgets.Display; |
| import org.eclipse.swt.widgets.Shell; |
| import org.eclipse.ui.handlers.HandlerUtil; |
| import org.eclipse.ui.texteditor.ITextEditor; |
| |
| /** |
| * Handler for command in charge of toggling comment prefixes. Based on |
| * org.eclipse.cdt.internal.ui.actions.ToggleCommentAction. |
| */ |
| public class ToggleCommentHandler extends AbstractHandler { |
| |
| /** The text operation target */ |
| private ITextOperationTarget operationTarget; |
| |
| /** |
| * Checks if the selected lines are all commented or not and |
| * uncomments/comments them respectively. |
| */ |
| @Override |
| public Object execute(ExecutionEvent event) { |
| ITextEditor editor = (ITextEditor) HandlerUtil.getActiveEditor(event); |
| if (editor == null || !editor.isEditable()) { |
| return null; |
| } |
| |
| updateOpTarget(editor); |
| if (operationTarget == null) { |
| return null; |
| } |
| |
| ISelection selection = editor.getSelectionProvider().getSelection(); |
| IDocument document = editor.getDocumentProvider().getDocument( |
| editor.getEditorInput()); |
| |
| final int operationCode; |
| if (isSelectionCommented(selection, document)) { |
| operationCode = ITextOperationTarget.STRIP_PREFIX; |
| } else { |
| operationCode = ITextOperationTarget.PREFIX; |
| } |
| |
| Shell shell = editor.getSite().getShell(); |
| if (!operationTarget.canDoOperation(operationCode)) { |
| if (shell != null) { |
| MessageDialog.openError(shell, |
| Localization.getString("ToggleComment_error_title"), //$NON-NLS-1$ |
| Localization.getString("ToggleComment_error_message")); //$NON-NLS-1$ |
| } |
| return null; |
| } |
| |
| Display display = null; |
| if (shell != null && !shell.isDisposed()) { |
| display = shell.getDisplay(); |
| } |
| |
| BusyIndicator.showWhile(display, new Runnable() { |
| @Override |
| public void run() { |
| operationTarget.doOperation(operationCode); |
| } |
| }); |
| |
| return null; |
| } |
| |
| /** |
| * Creates a region describing the text block (something that starts at the |
| * beginning of a line) completely containing the current selection. |
| * |
| * Note, the implementation has to match org.eclipse.jface.text.TextViewer; |
| * .getTextBlockFromSelection(). |
| * |
| * @param selection The selection to use |
| * @param document The document |
| * @return the region describing the text block comprising the given |
| * selection |
| * @throws BadLocationException |
| */ |
| public IRegion getTextBlockFromSelection(ITextSelection selection, |
| IDocument document) throws BadLocationException { |
| int start = document.getLineOffset(selection.getStartLine()); |
| int end; |
| int endLine = selection.getEndLine(); |
| if (document.getNumberOfLines() > endLine + 1) { |
| end = document.getLineOffset(endLine + 1); |
| } else { |
| end = document.getLength(); |
| } |
| return new Region(start, end - start); |
| } |
| |
| /** |
| * Is the given selection on the specified document single-line commented? |
| * |
| * @param selection Selection to check |
| * @param document The document |
| * @return <code>true</code> iff all selected lines are commented |
| */ |
| public boolean isSelectionCommented(ISelection selection, |
| IDocument document) { |
| |
| if (!(selection instanceof ITextSelection)) { |
| return false; |
| } |
| |
| ITextSelection textSelection = (ITextSelection) selection; |
| if (textSelection.getStartLine() < 0 || textSelection.getEndLine() < 0) { |
| return false; |
| } |
| |
| try { |
| IRegion block = getTextBlockFromSelection(textSelection, document); |
| ITypedRegion[] regions = TextUtilities.computePartitioning( |
| document, STPPartitionScanner.STP_PARTITIONING, |
| block.getOffset(), block.getLength(), false); |
| |
| int[] lines = new int[regions.length * 2]; // [startline, endline, |
| // startline, endline, |
| // ...] |
| |
| // For each partition in the text selection, figure out the |
| // startline and endline. |
| // Count the number of lines that are selected. |
| for (int i = 0, j = 0; i < regions.length; i++, j += 2) { |
| // Start line of region |
| lines[j] = getFirstCompleteLineOfRegion(regions[i], document); |
| // End line of region |
| int length = regions[i].getLength(); |
| int offset = regions[i].getOffset() + length; |
| if (length > 0) { |
| offset--; |
| } |
| |
| // If there is no startline for this region (startline = -1), |
| // then there is no endline, |
| // otherwise, get the line number of the endline and store it in |
| // the array. |
| lines[j + 1] = (lines[j] == -1 ? -1 : document |
| .getLineOfOffset(offset)); |
| |
| assert i < regions.length; |
| assert j < regions.length * 2; |
| } |
| |
| // Perform the check |
| boolean hasComment = false; |
| for (int i = 0, j = 0; i < regions.length; i++, j += 2) { |
| String prefix = "//"; //$NON-NLS-1$ |
| if (lines[j] >= 0 && lines[j + 1] >= 0) { |
| if (isBlockCommented(lines[j], lines[j + 1], prefix, |
| document)) { |
| hasComment = true; |
| } else if (!isBlockEmpty(lines[j], lines[j + 1], document)) { |
| return false; |
| } |
| } |
| } |
| return hasComment; |
| } catch (BadLocationException e) { |
| ExceptionErrorDialog.openError(e.getLocalizedMessage(), e); |
| } |
| |
| return false; |
| } |
| |
| /** |
| * Returns the index of the first line whose start offset is in the given |
| * text range. |
| * |
| * @param region the text range in characters where to find the line |
| * @param document The document |
| * @return the first line whose start index is in the given range, -1 if |
| * there is no such line |
| */ |
| public int getFirstCompleteLineOfRegion(IRegion region, IDocument document) { |
| try { |
| int startLine = document.getLineOfOffset(region.getOffset()); |
| |
| int offset = document.getLineOffset(startLine); |
| if (offset >= region.getOffset()) { |
| return startLine; |
| } |
| |
| offset = document.getLineOffset(startLine + 1); |
| return (offset > region.getOffset() + region.getLength() ? -1 |
| : startLine + 1); |
| } catch (BadLocationException e) { |
| ExceptionErrorDialog.openError(e.getLocalizedMessage(), e); |
| } |
| |
| return -1; |
| } |
| |
| /** |
| * Determines whether each line is empty |
| * |
| * @param startLine Start line in document |
| * @param endLine End line in document |
| * @param document The document |
| * @return <code>true</code> if each line from <code>startLine</code> to and |
| * including <code>endLine</code> is empty |
| */ |
| public boolean isBlockEmpty(int startLine, int endLine, IDocument document) { |
| try { |
| for (int i = startLine; i <= endLine; i++) { |
| IRegion line = document.getLineInformation(i); |
| String text = document.get(line.getOffset(), line.getLength()); |
| |
| boolean isEmptyLine = text.trim().length() == 0; |
| if (!isEmptyLine) { |
| return false; |
| } |
| } |
| return true; |
| } catch (BadLocationException e) { |
| ExceptionErrorDialog.openError(e.getLocalizedMessage(), e); |
| } |
| |
| return false; |
| } |
| |
| /** |
| * Determines whether each line is prefixed by one of the prefixes. |
| * |
| * @param startLine Start line in document |
| * @param endLine End line in document |
| * @param prefix Comment prefix |
| * @param document The document |
| * @return <code>true</code> iff each line from <code>startLine</code> to |
| * and including <code>endLine</code> is prepended by the |
| * <code>prefix</code>, ignoring whitespace at the begin of line |
| */ |
| public boolean isBlockCommented(int startLine, int endLine, String prefix, |
| IDocument document) { |
| try { |
| // Check for occurrences of prefixes in the given lines |
| boolean hasComment = false; |
| for (int i = startLine; i <= endLine; i++) { |
| IRegion line = document.getLineInformation(i); |
| String text = document.get(line.getOffset(), line.getLength()); |
| |
| boolean isEmptyLine = text.trim().length() == 0; |
| if (isEmptyLine) { |
| continue; |
| } |
| |
| int prefixIndex = text.indexOf(prefix, 0); |
| |
| if (prefixIndex == -1) { |
| // Found a line which is not commented |
| return false; |
| } |
| String s = document.get(line.getOffset(), prefixIndex); |
| s = s.trim(); |
| if (s.length() != 0) { |
| // Found a line which is not commented |
| return false; |
| } |
| hasComment = true; |
| } |
| return hasComment; |
| } catch (BadLocationException e) { |
| ExceptionErrorDialog.openError(e.getLocalizedMessage(), e); |
| } |
| |
| return false; |
| } |
| |
| /** |
| * Update text operation target based on the specified text editor. |
| * |
| * @param editor ITextEditor editor to associate operation target to. |
| */ |
| private void updateOpTarget(ITextEditor editor) { |
| if (editor != null) { |
| operationTarget = editor.getAdapter(ITextOperationTarget.class); |
| } |
| } |
| } |