| /******************************************************************************* |
| * Copyright (c) 2000, 2016 IBM Corporation 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: |
| * IBM Corporation - initial API and implementation |
| *******************************************************************************/ |
| package org.eclipse.jdt.internal.corext.util; |
| |
| import org.eclipse.osgi.util.TextProcessor; |
| |
| import org.eclipse.core.runtime.Assert; |
| |
| import org.eclipse.jface.action.LegacyActionTools; |
| import org.eclipse.jface.viewers.StyledString; |
| import org.eclipse.jface.viewers.StyledString.Styler; |
| |
| import org.eclipse.jface.text.BadLocationException; |
| import org.eclipse.jface.text.DefaultLineTracker; |
| import org.eclipse.jface.text.ILineTracker; |
| import org.eclipse.jface.text.IRegion; |
| |
| import org.eclipse.jdt.core.IJavaProject; |
| import org.eclipse.jdt.core.formatter.IndentManipulation; |
| |
| |
| /** |
| * Helper class to provide String manipulation functions not available in standard JDK. |
| */ |
| public class Strings { |
| |
| private Strings(){} |
| |
| |
| /** |
| * Tells whether we have to use the {@link TextProcessor} |
| * <p> |
| * This is used for performance optimization. |
| * </p> |
| * @since 3.4 |
| */ |
| public static final boolean USE_TEXT_PROCESSOR; |
| static { |
| String testString= "args : String[]"; //$NON-NLS-1$ |
| USE_TEXT_PROCESSOR= testString != TextProcessor.process(testString); |
| } |
| |
| private static final String JAVA_ELEMENT_DELIMITERS= TextProcessor.getDefaultDelimiters() + "<>(),?{} "; //$NON-NLS-1$ |
| |
| /** |
| * Sets the given <code>styler</code> to use for <code>matchingRegions</code> (obtained from |
| * {@link org.eclipse.jdt.core.search.SearchPattern#getMatchingRegions}) in the |
| * <code>styledString</code> starting from the given <code>index</code>. |
| * |
| * @param styledString the styled string to mark |
| * @param index the index from which to start marking |
| * @param matchingRegions the regions to mark |
| * @param styler the styler to use for marking |
| */ |
| public static void markMatchingRegions(StyledString styledString, int index, int[] matchingRegions, Styler styler) { |
| if (matchingRegions != null) { |
| int offset= -1; |
| int length= 0; |
| for (int i= 0; i + 1 < matchingRegions.length; i= i + 2) { |
| if (offset == -1) |
| offset= index + matchingRegions[i]; |
| |
| // Concatenate adjacent regions |
| if (i + 2 < matchingRegions.length && matchingRegions[i] + matchingRegions[i + 1] == matchingRegions[i + 2]) { |
| length= length + matchingRegions[i + 1]; |
| } else { |
| styledString.setStyle(offset, length + matchingRegions[i + 1], styler); |
| offset= -1; |
| length= 0; |
| } |
| } |
| } |
| } |
| |
| /** |
| * Adds special marks so that that the given styled string is readable in a BiDi environment. |
| * |
| * @param styledString the styled string |
| * @return the processed styled string |
| * @since 3.4 |
| */ |
| public static StyledString markLTR(StyledString styledString) { |
| |
| /* |
| * NOTE: For performance reasons we do not call markLTR(styledString, null) |
| */ |
| |
| if (!USE_TEXT_PROCESSOR) |
| return styledString; |
| |
| String inputString= styledString.getString(); |
| String string= TextProcessor.process(inputString); |
| if (string != inputString) |
| insertMarks(styledString, inputString, string); |
| return styledString; |
| } |
| |
| /** |
| * Adds special marks so that that the given styled Java element label is readable in a BiDi |
| * environment. |
| * |
| * @param styledString the styled string |
| * @return the processed styled string |
| * @since 3.6 |
| */ |
| public static StyledString markJavaElementLabelLTR(StyledString styledString) { |
| if (!USE_TEXT_PROCESSOR) |
| return styledString; |
| |
| String inputString= styledString.getString(); |
| String string= TextProcessor.process(inputString, JAVA_ELEMENT_DELIMITERS); |
| if (string != inputString) |
| insertMarks(styledString, inputString, string); |
| return styledString; |
| } |
| |
| /** |
| * Adds special marks so that that the given styled string is readable in a BiDi environment. |
| * |
| * @param styledString the styled string |
| * @param delimiters the additional delimiters |
| * @return the processed styled string |
| * @since 3.4 |
| */ |
| public static StyledString markLTR(StyledString styledString, String delimiters) { |
| if (!USE_TEXT_PROCESSOR) |
| return styledString; |
| |
| String inputString= styledString.getString(); |
| String string= TextProcessor.process(inputString, delimiters); |
| if (string != inputString) |
| insertMarks(styledString, inputString, string); |
| return styledString; |
| } |
| |
| /** |
| * Inserts the marks into the given styled string. |
| * |
| * @param styledString the styled string |
| * @param originalString the original string |
| * @param processedString the processed string |
| * @since 3.5 |
| */ |
| private static void insertMarks(StyledString styledString, String originalString, String processedString) { |
| int originalLength= originalString.length(); |
| int processedStringLength= processedString.length(); |
| char orig= originalLength > 0 ? originalString.charAt(0) : '\0'; |
| for (int o= 0, p= 0; p < processedStringLength; p++) { |
| char processed= processedString.charAt(p); |
| if (o < originalLength) { |
| if (orig == processed) { |
| o++; |
| if (o < originalLength) |
| orig= originalString.charAt(o); |
| continue; |
| } |
| } |
| styledString.insert(processed, p); |
| } |
| } |
| |
| /** |
| * Adds special marks so that that the given string is readable in a BiDi environment. |
| * |
| * @param string the string |
| * @return the processed styled string |
| * @since 3.4 |
| */ |
| public static String markLTR(String string) { |
| if (!USE_TEXT_PROCESSOR) |
| return string; |
| |
| return TextProcessor.process(string); |
| } |
| |
| /** |
| * Adds special marks so that that the given string is readable in a BiDi environment. |
| * |
| * @param string the string |
| * @param delimiters the delimiters |
| * @return the processed styled string |
| * @since 3.4 |
| */ |
| public static String markLTR(String string, String delimiters) { |
| if (!USE_TEXT_PROCESSOR) |
| return string; |
| |
| return TextProcessor.process(string, delimiters); |
| } |
| |
| /** |
| * Adds special marks so that that the given Java element label is readable in a BiDi |
| * environment. |
| * |
| * @param string the string |
| * @return the processed styled string |
| * @since 3.6 |
| */ |
| public static String markJavaElementLabelLTR(String string) { |
| if (!USE_TEXT_PROCESSOR) |
| return string; |
| |
| return TextProcessor.process(string, JAVA_ELEMENT_DELIMITERS); |
| } |
| |
| /** |
| * Tests if a char is lower case. Fix for 26529. |
| * @param ch the char |
| * @return return true if char is lower case |
| */ |
| public static boolean isLowerCase(char ch) { |
| return Character.toLowerCase(ch) == ch; |
| } |
| |
| public static boolean startsWithIgnoreCase(String text, String prefix) { |
| int textLength= text.length(); |
| int prefixLength= prefix.length(); |
| if (textLength < prefixLength) |
| return false; |
| for (int i= prefixLength - 1; i >= 0; i--) { |
| if (Character.toLowerCase(prefix.charAt(i)) != Character.toLowerCase(text.charAt(i))) |
| return false; |
| } |
| return true; |
| } |
| |
| public static String removeNewLine(String message) { |
| StringBuffer result= new StringBuffer(); |
| int current= 0; |
| int index= message.indexOf('\n', 0); |
| while (index != -1) { |
| result.append(message.substring(current, index)); |
| if (current < index && index != 0) |
| result.append(' '); |
| current= index + 1; |
| index= message.indexOf('\n', current); |
| } |
| result.append(message.substring(current)); |
| return result.toString(); |
| } |
| |
| /** |
| * Converts the given string into an array of lines. The lines |
| * don't contain any line delimiter characters. |
| * |
| * @param input the string |
| * @return the string converted into an array of strings. Returns <code> |
| * null</code> if the input string can't be converted in an array of lines. |
| */ |
| public static String[] convertIntoLines(String input) { |
| try { |
| ILineTracker tracker= new DefaultLineTracker(); |
| tracker.set(input); |
| int size= tracker.getNumberOfLines(); |
| String result[]= new String[size]; |
| for (int i= 0; i < size; i++) { |
| IRegion region= tracker.getLineInformation(i); |
| int offset= region.getOffset(); |
| result[i]= input.substring(offset, offset + region.getLength()); |
| } |
| return result; |
| } catch (BadLocationException e) { |
| return null; |
| } |
| } |
| |
| /** |
| * Returns <code>true</code> if the given string only consists of |
| * white spaces according to Java. If the string is empty, <code>true |
| * </code> is returned. |
| * |
| * @param s the string to test |
| * @return <code>true</code> if the string only consists of white |
| * spaces; otherwise <code>false</code> is returned |
| * |
| * @see java.lang.Character#isWhitespace(char) |
| */ |
| public static boolean containsOnlyWhitespaces(String s) { |
| int size= s.length(); |
| for (int i= 0; i < size; i++) { |
| if (!Character.isWhitespace(s.charAt(i))) |
| return false; |
| } |
| return true; |
| } |
| |
| /** |
| * Removes leading tabs and spaces from the given string. If the string |
| * doesn't contain any leading tabs or spaces then the string itself is |
| * returned. |
| * @param line the line |
| * @return the trimmed line |
| */ |
| public static String trimLeadingTabsAndSpaces(String line) { |
| int size= line.length(); |
| int start= size; |
| for (int i= 0; i < size; i++) { |
| char c= line.charAt(i); |
| if (!IndentManipulation.isIndentChar(c)) { |
| start= i; |
| break; |
| } |
| } |
| if (start == 0) |
| return line; |
| else if (start == size) |
| return ""; //$NON-NLS-1$ |
| else |
| return line.substring(start); |
| } |
| |
| public static String trimTrailingTabsAndSpaces(String line) { |
| int size= line.length(); |
| int end= size; |
| for (int i= size - 1; i >= 0; i--) { |
| char c= line.charAt(i); |
| if (IndentManipulation.isIndentChar(c)) { |
| end= i; |
| } else { |
| break; |
| } |
| } |
| if (end == size) |
| return line; |
| else if (end == 0) |
| return ""; //$NON-NLS-1$ |
| else |
| return line.substring(0, end); |
| } |
| |
| /** |
| * Returns the indent of the given string in indentation units. Odd spaces |
| * are not counted. |
| * |
| * @param line the text line |
| * @param project the java project from which to get the formatter |
| * preferences, or <code>null</code> for global preferences |
| * @return the number of indent units |
| * @since 3.1 |
| */ |
| public static int computeIndentUnits(String line, IJavaProject project) { |
| return IndentManipulation.measureIndentUnits(line, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project)); |
| } |
| |
| /** |
| * Returns the indent of the given string in indentation units. Odd spaces |
| * are not counted. |
| * |
| * @param line the text line |
| * @param tabWidth the width of the '\t' character in space equivalents |
| * @param indentWidth the width of one indentation unit in space equivalents |
| * @return the indent of the given string in indentation units |
| * |
| * @since 3.1 |
| */ |
| public static int computeIndentUnits(String line, int tabWidth, int indentWidth) { |
| return IndentManipulation.measureIndentUnits(line, tabWidth, indentWidth); |
| } |
| |
| /** |
| * Computes the visual length of the indentation of a |
| * <code>CharSequence</code>, counting a tab character as the size until |
| * the next tab stop and every other whitespace character as one. |
| * |
| * @param line the string to measure the indent of |
| * @param tabSize the visual size of a tab in space equivalents |
| * @return the visual length of the indentation of <code>line</code> |
| * @since 3.1 |
| */ |
| public static int measureIndentLength(CharSequence line, int tabSize) { |
| return IndentManipulation.measureIndentInSpaces(line, tabSize); |
| } |
| |
| /** |
| * Removes the given number of indents from the line. Asserts that the given line |
| * has the requested number of indents. If <code>indentsToRemove <= 0</code> |
| * the line is returned. |
| * @param line the line |
| * @param indentsToRemove the indents to remove |
| * |
| * @param project the java project from which to get the formatter |
| * preferences, or <code>null</code> for global preferences |
| * @return the trimmed line |
| * |
| * @since 3.1 |
| */ |
| public static String trimIndent(String line, int indentsToRemove, IJavaProject project) { |
| return IndentManipulation.trimIndent(line, indentsToRemove, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project)); |
| } |
| |
| /** |
| * Removes the given number of indents from the line. Asserts that the given line |
| * has the requested number of indents. If <code>indentsToRemove <= 0</code> |
| * the line is returned. |
| * @param line the line |
| * @param indentsToRemove the indents to remove |
| * @param tabWidth the tab width |
| * @param indentWidth the indent width |
| * @return the trimmed line |
| * |
| * @since 3.1 |
| */ |
| public static String trimIndent(String line, int indentsToRemove, int tabWidth, int indentWidth) { |
| return IndentManipulation.trimIndent(line, indentsToRemove, tabWidth, indentWidth); |
| } |
| |
| /** |
| * Removes the common number of indents from all lines. If a line |
| * only consists out of white space it is ignored. |
| * @param lines the lines |
| |
| * @param project the java project from which to get the formatter |
| * preferences, or <code>null</code> for global preferences |
| * @since 3.1 |
| */ |
| public static void trimIndentation(String[] lines, IJavaProject project) { |
| trimIndentation(lines, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project), true); |
| } |
| /** |
| * Removes the common number of indents from all lines. If a line |
| * only consists out of white space it is ignored. |
| * @param lines the lines |
| * @param tabWidth the size of one tab in space equivalents |
| * @param indentWidth the size of the indent in space equivalents |
| * |
| * @since 3.1 |
| */ |
| public static void trimIndentation(String[] lines, int tabWidth, int indentWidth) { |
| trimIndentation(lines, tabWidth, indentWidth, true); |
| } |
| |
| /** |
| * Removes the common number of indents from all lines. If a line |
| * only consists out of white space it is ignored. If <code> |
| * considerFirstLine</code> is false the first line will be ignored. |
| * @param lines the lines |
| * |
| * @param project the java project from which to get the formatter |
| * preferences, or <code>null</code> for global preferences |
| * @param considerFirstLine If <code>considerFirstLine</code> is false the first line will be ignored. |
| * @since 3.1 |
| */ |
| public static void trimIndentation(String[] lines, IJavaProject project, boolean considerFirstLine) { |
| trimIndentation(lines, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project), considerFirstLine); |
| } |
| |
| /** |
| * Removes the common number of indents from all lines. If a line |
| * only consists out of white space it is ignored. If <code> |
| * considerFirstLine</code> is false the first line will be ignored. |
| * @param lines the lines |
| * @param tabWidth the size of one tab in space equivalents |
| * @param indentWidth the size of the indent in space equivalents |
| * @param considerFirstLine If <code> considerFirstLine</code> is false the first line will be ignored. |
| * @since 3.1 |
| */ |
| public static void trimIndentation(String[] lines, int tabWidth, int indentWidth, boolean considerFirstLine) { |
| String[] toDo= new String[lines.length]; |
| // find indentation common to all lines |
| int minIndent= Integer.MAX_VALUE; // very large |
| for (int i= considerFirstLine ? 0 : 1; i < lines.length; i++) { |
| String line= lines[i]; |
| if (containsOnlyWhitespaces(line)) |
| continue; |
| toDo[i]= line; |
| int indent= computeIndentUnits(line, tabWidth, indentWidth); |
| if (indent < minIndent) { |
| minIndent= indent; |
| } |
| } |
| |
| if (minIndent > 0) { |
| // remove this indent from all lines |
| for (int i= considerFirstLine ? 0 : 1; i < toDo.length; i++) { |
| String s= toDo[i]; |
| if (s != null) |
| lines[i]= trimIndent(s, minIndent, tabWidth, indentWidth); |
| else { |
| String line= lines[i]; |
| int indent= computeIndentUnits(line, tabWidth, indentWidth); |
| if (indent > minIndent) |
| lines[i]= trimIndent(line, minIndent, tabWidth, indentWidth); |
| else |
| lines[i]= trimLeadingTabsAndSpaces(line); |
| } |
| } |
| } |
| } |
| |
| /** |
| * Returns that part of the indentation of <code>line</code> that makes up |
| * a multiple of indentation units. |
| * |
| * @param line the line to scan |
| * @param project the java project from which to get the formatter |
| * preferences, or <code>null</code> for global preferences |
| * @return the indent part of <code>line</code>, but no odd spaces |
| * @since 3.1 |
| */ |
| public static String getIndentString(String line, IJavaProject project) { |
| return IndentManipulation.extractIndentString(line, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project)); |
| } |
| |
| /** |
| * Returns that part of the indentation of <code>line</code> that makes up |
| * a multiple of indentation units. |
| * |
| * @param line the line to scan |
| * @param tabWidth the size of one tab in space equivalents |
| * @param indentWidth the size of the indent in space equivalents |
| * @return the indent part of <code>line</code>, but no odd spaces |
| * @since 3.1 |
| */ |
| public static String getIndentString(String line, int tabWidth, int indentWidth) { |
| return IndentManipulation.extractIndentString(line, tabWidth, indentWidth); |
| } |
| |
| public static String[] removeTrailingEmptyLines(String[] sourceLines) { |
| int lastNonEmpty= findLastNonEmptyLineIndex(sourceLines); |
| String[] result= new String[lastNonEmpty + 1]; |
| for (int i= 0; i < result.length; i++) { |
| result[i]= sourceLines[i]; |
| } |
| return result; |
| } |
| |
| private static int findLastNonEmptyLineIndex(String[] sourceLines) { |
| for (int i= sourceLines.length - 1; i >= 0; i--) { |
| if (! sourceLines[i].trim().equals(""))//$NON-NLS-1$ |
| return i; |
| } |
| return -1; |
| } |
| |
| /** |
| * Change the indent of, possible multi-line, code range. The current indent is removed, a new indent added. |
| * The first line of the code will not be changed. (It is considered to have no indent as it might start in |
| * the middle of a line) |
| * @param code the code |
| * @param codeIndentLevel level of indentation |
| * |
| * @param project the java project from which to get the formatter |
| * preferences, or <code>null</code> for global preferences |
| * @param newIndent new indent |
| * @param lineDelim line delimiter |
| * @return the changed code |
| * @since 3.1 |
| */ |
| public static String changeIndent(String code, int codeIndentLevel, IJavaProject project, String newIndent, String lineDelim) { |
| return IndentManipulation.changeIndent(code, codeIndentLevel, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project), newIndent, lineDelim); |
| } |
| |
| /** |
| * Change the indent of, possible muti-line, code range. The current indent is removed, a new indent added. |
| * The first line of the code will not be changed. (It is considered to have no indent as it might start in |
| * the middle of a line) |
| * @param code the code |
| * @param codeIndentLevel indent level |
| * @param tabWidth the size of one tab in space equivalents |
| * @param indentWidth the size of the indent in space equivalents |
| * @param newIndent new indent |
| * @param lineDelim line delimiter |
| * @return the changed code |
| * @since 3.1 |
| */ |
| public static String changeIndent(String code, int codeIndentLevel, int tabWidth, int indentWidth, String newIndent, String lineDelim) { |
| return IndentManipulation.changeIndent(code, codeIndentLevel, tabWidth, indentWidth, newIndent, lineDelim); |
| } |
| |
| public static String trimIndentation(String source, IJavaProject project, boolean considerFirstLine) { |
| return trimIndentation(source, CodeFormatterUtil.getTabWidth(project), CodeFormatterUtil.getIndentWidth(project), considerFirstLine); |
| } |
| |
| public static String trimIndentation(String source, int tabWidth, int indentWidth, boolean considerFirstLine) { |
| try { |
| ILineTracker tracker= new DefaultLineTracker(); |
| tracker.set(source); |
| int size= tracker.getNumberOfLines(); |
| if (size == 1) |
| return source; |
| String lines[]= new String[size]; |
| for (int i= 0; i < size; i++) { |
| IRegion region= tracker.getLineInformation(i); |
| int offset= region.getOffset(); |
| lines[i]= source.substring(offset, offset + region.getLength()); |
| } |
| Strings.trimIndentation(lines, tabWidth, indentWidth, considerFirstLine); |
| StringBuffer result= new StringBuffer(); |
| int last= size - 1; |
| for (int i= 0; i < size; i++) { |
| result.append(lines[i]); |
| if (i < last) |
| result.append(tracker.getLineDelimiter(i)); |
| } |
| return result.toString(); |
| } catch (BadLocationException e) { |
| Assert.isTrue(false,"Can not happend"); //$NON-NLS-1$ |
| return null; |
| } |
| } |
| |
| |
| /** |
| * Concatenate the given strings into one strings using the passed line delimiter as a |
| * delimiter. No delimiter is added to the last line. |
| * @param lines the lines |
| * @param delimiter line delimiter |
| * @return the concatenated lines |
| */ |
| public static String concatenate(String[] lines, String delimiter) { |
| StringBuffer buffer= new StringBuffer(); |
| for (int i= 0; i < lines.length; i++) { |
| if (i > 0) |
| buffer.append(delimiter); |
| buffer.append(lines[i]); |
| } |
| return buffer.toString(); |
| } |
| |
| public static boolean equals(String s, char[] c) { |
| if (s.length() != c.length) |
| return false; |
| |
| for (int i = c.length; --i >= 0;) |
| if (s.charAt(i) != c[i]) |
| return false; |
| return true; |
| } |
| |
| public static String removeTrailingCharacters(String text, char toRemove) { |
| int size= text.length(); |
| int end= size; |
| for (int i= size - 1; i >= 0; i--) { |
| char c= text.charAt(i); |
| if (c == toRemove) { |
| end= i; |
| } else { |
| break; |
| } |
| } |
| if (end == size) |
| return text; |
| else if (end == 0) |
| return ""; //$NON-NLS-1$ |
| else |
| return text.substring(0, end); |
| } |
| |
| public static String removeMnemonicIndicator(String string) { |
| return LegacyActionTools.removeMnemonics(string); |
| } |
| } |