1 /*******************************************************************************
2 * Copyright (c) 2000, 2001, 2002 International Business Machines Corp. and others.
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Common Public License v0.5
5 * which accompanies this distribution, and is available at
6 * http://www.eclipse.org/legal/cpl-v05.html
9 * IBM Corporation - initial API and implementation
10 ******************************************************************************/
11 package net.sourceforge.phpdt.internal.formatter;
13 import java.io.BufferedReader;
14 import java.io.IOException;
15 import java.io.StringReader;
16 import java.util.Hashtable;
17 import java.util.Locale;
20 import net.sourceforge.phpdt.core.ICodeFormatter;
21 import net.sourceforge.phpdt.core.compiler.CharOperation;
22 import net.sourceforge.phpdt.core.compiler.ITerminalSymbols;
23 import net.sourceforge.phpdt.core.compiler.InvalidInputException;
24 import net.sourceforge.phpdt.internal.compiler.ConfigurableOption;
25 import net.sourceforge.phpdt.internal.compiler.parser.Scanner;
26 import net.sourceforge.phpdt.internal.corext.codemanipulation.StubUtility;
27 import net.sourceforge.phpdt.internal.corext.util.Strings;
28 import net.sourceforge.phpdt.internal.formatter.impl.FormatterOptions;
29 import net.sourceforge.phpdt.internal.formatter.impl.SplitLine;
30 import net.sourceforge.phpdt.internal.ui.preferences.CodeFormatterPreferencePage;
32 import org.eclipse.jface.text.IDocument;
33 import org.eclipse.jface.text.formatter.IContentFormatterExtension;
34 import org.eclipse.jface.text.formatter.IFormattingContext;
37 * <h2>How to format a piece of code ?</h2>
39 * <li>Create an instance of <code>CodeFormatter</code>
40 * <li>Use the method <code>void format(aString)</code> on this instance to format <code>aString</code>. It will return the
44 public class CodeFormatter implements ITerminalSymbols, ICodeFormatter {
45 // IContentFormatterExtension {
46 public FormatterOptions options;
49 * Represents a block in the <code>constructions</code> stack.
51 public static final int BLOCK = ITerminalSymbols.TokenNameLBRACE;
54 * Represents a block following a control statement in the <code>constructions</code> stack.
56 public static final int NONINDENT_BLOCK = -100;
59 * Contains the formatted output.
61 StringBuffer formattedSource;
64 * Contains the current line. <br>
65 * Will be dumped at the next "newline"
67 StringBuffer currentLineBuffer;
70 * Used during the formatting to get each token.
75 * Contains the tokens responsible for the current indentation level and the blocks not closed yet.
77 private int[] constructions;
80 * Index in the <code>constructions</code> array.
82 private int constructionsCount;
85 * Level of indentation of the current token (number of tab char put in front of it).
87 private int indentationLevel;
90 * Regular level of indentation of all the lines
92 private int initialIndentationLevel;
95 * Used to split a line.
100 * To remember the offset between the beginning of the line and the beginning of the comment.
102 int currentCommentOffset;
104 int currentLineIndentationLevel;
106 int maxLineSize = 30;
108 private boolean containsOpenCloseBraces;
110 private int indentationLevelForOpenCloseBraces;
113 * Collections of positions to map
115 private int[] positionsToMap;
118 * Collections of mapped positions
120 private int[] mappedPositions;
122 private int indexToMap;
124 private int indexInMap;
126 private int globalDelta;
128 private int lineDelta;
130 private int splitDelta;
132 private int beginningOfLineIndex;
134 private int multipleLineCommentCounter;
137 * Creates a new instance of Code Formatter using the given settings.
139 * @deprecated backport 1.0 internal functionality
141 public CodeFormatter(ConfigurableOption[] settings) {
142 this(convertConfigurableOptions(settings));
146 * Creates a new instance of Code Formatter using the FormattingOptions object given as argument
148 * @deprecated Use CodeFormatter(ConfigurableOption[]) instead
150 public CodeFormatter() {
155 * Creates a new instance of Code Formatter using the given settings.
157 public CodeFormatter(Map settings) {
158 // initialize internal state
159 constructionsCount = 0;
160 constructions = new int[10];
161 currentLineIndentationLevel = indentationLevel = initialIndentationLevel;
162 currentCommentOffset = -1;
163 // initialize primary and secondary scanners
164 scanner = new Scanner(true /* comment */
165 , true /* whitespace */
168 , true, /* tokenizeStrings */
169 null, null); // regular scanner for forming lines
170 scanner.recordLineSeparator = true;
171 // to remind of the position of the beginning of the line.
172 splitScanner = new Scanner(true /* comment */
173 , true /* whitespace */
176 , true, /* tokenizeStrings */
178 // secondary scanner to split long lines formed by primary scanning
179 // initialize current line buffer
180 currentLineBuffer = new StringBuffer();
181 this.options = new FormatterOptions(settings);
185 * Returns true if a lineSeparator has to be inserted before <code>operator</code> false otherwise.
187 private static boolean breakLineBeforeOperator(int operator) {
190 case TokenNameSEMICOLON:
199 * @deprecated backport 1.0 internal functionality
201 private static Map convertConfigurableOptions(ConfigurableOption[] settings) {
202 Hashtable options = new Hashtable(10);
203 for (int i = 0; i < settings.length; i++) {
204 if (settings[i].getComponentName().equals(CodeFormatter.class.getName())) {
205 String optionName = settings[i].getOptionName();
206 int valueIndex = settings[i].getCurrentValueIndex();
207 if (optionName.equals("newline.openingBrace")) { //$NON-NLS-1$
208 options.put("net.sourceforge.phpdt.core.formatter.newline.openingBrace", valueIndex == 0 ? "insert" : "do not insert"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
209 } else if (optionName.equals("newline.controlStatement")) { //$NON-NLS-1$
211 .put("net.sourceforge.phpdt.core.formatter.newline.controlStatement", valueIndex == 0 ? "insert" : "do not insert"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
212 } else if (optionName.equals("newline.clearAll")) { //$NON-NLS-1$
213 options.put("net.sourceforge.phpdt.core.formatter.newline.clearAll", valueIndex == 0 ? "clear all" : "preserve one"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
214 } else if (optionName.equals("newline.elseIf")) { //$NON-NLS-1$
215 options.put("net.sourceforge.phpdt.core.formatter.newline.elseIf", valueIndex == 0 ? "do not insert" : "insert"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
216 } else if (optionName.equals("newline.emptyBlock")) { //$NON-NLS-1$
217 options.put("net.sourceforge.phpdt.core.formatter.newline.emptyBlock", valueIndex == 0 ? "insert" : "do not insert"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
218 } else if (optionName.equals("lineSplit")) { //$NON-NLS-1$
219 options.put("net.sourceforge.phpdt.core.formatter.lineSplit", String.valueOf(valueIndex)); //$NON-NLS-1$ //$NON-NLS-2$
220 } else if (optionName.equals("style.assignment")) { //$NON-NLS-1$
221 options.put("net.sourceforge.phpdt.core.formatter.style.assignment", valueIndex == 0 ? "compact" : "normal"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
222 } else if (optionName.equals("tabulation.char")) { //$NON-NLS-1$
223 options.put("net.sourceforge.phpdt.core.formatter.tabulation.char", valueIndex == 0 ? "tab" : "space"); //$NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
224 } else if (optionName.equals("tabulation.size")) { //$NON-NLS-1$
225 options.put("net.sourceforge.phpdt.core.formatter.tabulation.size", String.valueOf(valueIndex)); //$NON-NLS-1$ //$NON-NLS-2$
233 * Returns the end of the source code.
235 private final String copyRemainingSource() {
236 char str[] = scanner.source;
237 int startPosition = scanner.startPosition;
238 int length = str.length - startPosition;
239 StringBuffer bufr = new StringBuffer(length);
240 if (startPosition < str.length) {
241 bufr.append(str, startPosition, length);
243 return (bufr.toString());
247 * Inserts <code>tabCount</code> tab character or their equivalent number of spaces.
249 private void dumpTab(int tabCount) {
250 if (options.indentWithTab) {
251 for (int j = 0; j < tabCount; j++) {
252 formattedSource.append('\t');
253 increaseSplitDelta(1);
256 for (int i = 0, max = options.tabSize * tabCount; i < max; i++) {
257 formattedSource.append(' ');
258 increaseSplitDelta(1);
264 * Dumps <code>currentLineBuffer</code> into the formatted string.
266 private void flushBuffer() {
267 String currentString = currentLineBuffer.toString();
269 beginningOfLineIndex = formattedSource.length();
270 if (containsOpenCloseBraces) {
271 containsOpenCloseBraces = false;
272 outputLine(currentString, false, indentationLevelForOpenCloseBraces, 0, -1, null, 0);
273 indentationLevelForOpenCloseBraces = currentLineIndentationLevel;
275 outputLine(currentString, false, currentLineIndentationLevel, 0, -1, null, 0);
277 int scannerSourceLength = scanner.source.length;
278 if (scannerSourceLength > 2) {
279 if (scanner.source[scannerSourceLength - 1] == '\n' && scanner.source[scannerSourceLength - 2] == '\r') {
280 formattedSource.append(options.lineSeparatorSequence);
281 increaseGlobalDelta(options.lineSeparatorSequence.length - 2);
282 } else if (scanner.source[scannerSourceLength - 1] == '\n') {
283 formattedSource.append(options.lineSeparatorSequence);
284 increaseGlobalDelta(options.lineSeparatorSequence.length - 1);
285 } else if (scanner.source[scannerSourceLength - 1] == '\r') {
286 formattedSource.append(options.lineSeparatorSequence);
287 increaseGlobalDelta(options.lineSeparatorSequence.length - 1);
290 updateMappedPositions(scanner.startPosition);
294 * Formats the input string.
296 private void format() {
298 int previousToken = 0;
299 int previousCompilableToken = 0;
300 int indentationOffset = 0;
301 int newLinesInWhitespace = 0;
302 // number of new lines in the previous whitespace token
303 // (used to leave blank lines before comments)
304 int pendingNewLines = 0;
305 boolean expectingOpenBrace = false;
306 boolean clearNonBlockIndents = false;
307 // true if all indentations till the 1st { (usefull after } or ;)
308 boolean pendingSpace = true;
309 boolean pendingNewlineAfterParen = false;
310 // true when a cr is to be put after a ) (in conditional statements)
311 boolean inAssignment = false;
312 boolean inArrayAssignment = false;
313 boolean inThrowsClause = false;
314 boolean inClassOrInterfaceHeader = false;
315 int dollarBraceCount = 0;
316 // openBracketCount is used to count the number of open brackets not closed
318 int openBracketCount = 0;
319 int unarySignModifier = 0;
320 // openParenthesis[0] is used to count the parenthesis not belonging to a
322 // (eg foo();). parenthesis in for (...) are count elsewhere in the array.
323 int openParenthesisCount = 1;
324 int[] openParenthesis = new int[10];
325 // tokenBeforeColon is used to know what token goes along with the current
327 // it can be case or ?
328 int tokenBeforeColonCount = 0;
329 int[] tokenBeforeColon = new int[10];
330 constructionsCount = 0; // initializes the constructions count.
331 // contains DO if in a DO..WHILE statement, UNITIALIZED otherwise.
333 // fix for 1FF17XY: LFCOM:ALL - Format problem on not matching } and else
334 boolean specialElse = false;
335 // OPTION (IndentationLevel): initial indentation level may be non-zero.
336 currentLineIndentationLevel += constructionsCount;
337 // An InvalidInputException exception might cause the termination of this
341 // Get the next token. Catch invalid input and output it
342 // with minimal formatting, also catch end of input and
345 token = scanner.getNextToken();
347 int currentEndPosition = scanner.getCurrentTokenEndPosition();
348 int currentStartPosition = scanner.getCurrentTokenStartPosition();
349 System.out.print(currentStartPosition + "," + currentEndPosition + ": ");
350 System.out.println(scanner.toStringAction(token));
352 // Patch for line comment
353 // See PR http://dev.eclipse.org/bugs/show_bug.cgi?id=23096
354 if (token == ITerminalSymbols.TokenNameCOMMENT_LINE) {
355 int length = scanner.currentPosition;
356 loop: for (int index = length - 1; index >= 0; index--) {
357 switch (scanner.source[index]) {
360 scanner.currentPosition--;
367 } catch (InvalidInputException e) {
368 if (!handleInvalidToken(e)) {
373 if (token == Scanner.TokenNameEOF)
376 * ## MODIFYING the indentation level before generating new lines and indentation in the output string
378 // Removes all the indentations made by statements not followed by a
380 // except if the current token is ELSE, CATCH or if we are in a
382 if (clearNonBlockIndents && (token != Scanner.TokenNameWHITESPACE)) {
385 if (constructionsCount > 0 && constructions[constructionsCount - 1] == TokenNameelse) {
389 indentationLevel += popInclusiveUntil(TokenNameif);
391 // case TokenNamecatch :
392 // indentationLevel += popInclusiveUntil(TokenNamecatch);
394 // case TokenNamefinally :
395 // indentationLevel += popInclusiveUntil(TokenNamecatch);
398 if (nlicsToken == TokenNamedo) {
399 indentationLevel += pop(TokenNamedo);
403 indentationLevel += popExclusiveUntilBlockOrCase();
404 // clear until a CASE, DEFAULT or BLOCK is encountered.
405 // Thus, the indentationLevel is correctly cleared either
406 // in a switch/case statement or in any other situation.
408 clearNonBlockIndents = false;
410 // returns to the indentation level created by the SWITCH keyword
411 // if the current token is a CASE or a DEFAULT
412 if (token == TokenNamecase || token == TokenNamedefault) {
413 indentationLevel += pop(TokenNamecase);
415 // if (token == Scanner.TokenNamethrows) {
416 // inThrowsClause = true;
418 if ((token == Scanner.TokenNameclass || token == Scanner.TokenNameinterface) && previousToken != Scanner.TokenNameDOT) {
419 inClassOrInterfaceHeader = true;
422 * ## APPEND newlines and indentations to the output string
424 // Do not add a new line between ELSE and IF, if the option
425 // elseIfOnSameLine is true.
426 // Fix for 1ETLWPZ: IVJCOM:ALL - incorrect "else if" formatting
427 // if (pendingNewlineAfterParen
428 // && previousCompilableToken == TokenNameelse
429 // && token == TokenNameif
430 // && options.compactElseIfMode) {
431 // pendingNewlineAfterParen = false;
432 // pendingNewLines = 0;
433 // indentationLevel += pop(TokenNameelse);
434 // // because else if is now one single statement,
435 // // the indentation level after it is increased by one and not by 2
436 // // (else = 1 indent, if = 1 indent, but else if = 1 indent, not 2).
438 // Add a newline & indent to the formatted source string if
439 // a for/if-else/while statement was scanned and there is no block
441 pendingNewlineAfterParen = pendingNewlineAfterParen
442 || (previousCompilableToken == TokenNameRPAREN && token == TokenNameLBRACE);
443 if (pendingNewlineAfterParen && token != Scanner.TokenNameWHITESPACE) {
444 pendingNewlineAfterParen = false;
445 // Do to add a newline & indent sequence if the current token is an
446 // open brace or a period or if the current token is a semi-colon and
448 // previous token is a close paren.
449 // add a new line if a parenthesis belonging to a for() statement
450 // has been closed and the current token is not an opening brace
451 if (token != TokenNameLBRACE && !isComment(token)
452 // to avoid adding new line between else and a comment
453 && token != TokenNameDOT && !(previousCompilableToken == TokenNameRPAREN && token == TokenNameSEMICOLON)) {
455 currentLineIndentationLevel = indentationLevel;
457 pendingSpace = false;
459 if (token == TokenNameLBRACE && options.newLineBeforeOpeningBraceMode) {
461 if (constructionsCount > 0 && constructions[constructionsCount - 1] != BLOCK
462 && constructions[constructionsCount - 1] != NONINDENT_BLOCK) {
463 currentLineIndentationLevel = indentationLevel - 1;
465 currentLineIndentationLevel = indentationLevel;
468 pendingSpace = false;
472 if (token == TokenNameLBRACE && options.newLineBeforeOpeningBraceMode && constructionsCount > 0
473 && constructions[constructionsCount - 1] == TokenNamedo) {
475 currentLineIndentationLevel = indentationLevel - 1;
477 pendingSpace = false;
480 if (token == TokenNameLBRACE && inThrowsClause) {
481 inThrowsClause = false;
482 if (options.newLineBeforeOpeningBraceMode) {
484 currentLineIndentationLevel = indentationLevel;
486 pendingSpace = false;
490 if (token == TokenNameLBRACE && inClassOrInterfaceHeader) {
491 inClassOrInterfaceHeader = false;
492 if (options.newLineBeforeOpeningBraceMode) {
494 currentLineIndentationLevel = indentationLevel;
496 pendingSpace = false;
499 // Add pending new lines to the formatted source string.
500 // Note: pending new lines are not added if the current token
501 // is a single line comment or whitespace.
502 // if the comment is between parenthesis, there is no blank line
504 // (if it's a one-line comment, a blank line is added after it).
505 if (((pendingNewLines > 0 && (!isComment(token)))
506 || (newLinesInWhitespace > 0 && (openParenthesisCount <= 1 && isComment(token))) || (previousCompilableToken == TokenNameLBRACE && token == TokenNameRBRACE))
507 && token != Scanner.TokenNameWHITESPACE) {
508 // Do not add newline & indent between an adjoining close brace and
509 // close paren. Anonymous inner classes may use this form.
510 boolean closeBraceAndCloseParen = previousToken == TokenNameRBRACE && token == TokenNameRPAREN;
511 // OPTION (NewLineInCompoundStatement): do not add newline & indent
512 // between close brace and else, (do) while, catch, and finally if
513 // newlineInCompoundStatement is true.
514 boolean nlicsOption = previousToken == TokenNameRBRACE
515 && !options.newlineInControlStatementMode
516 && (token == TokenNameelse || (token == TokenNamewhile && nlicsToken == TokenNamedo) || token == TokenNamecatch || token == TokenNamefinally);
517 // Do not add a newline & indent between a close brace and
519 boolean semiColonAndCloseBrace = previousToken == TokenNameRBRACE && token == TokenNameSEMICOLON;
520 // Do not add a new line & indent between a multiline comment and a
522 boolean commentAndOpenBrace = previousToken == Scanner.TokenNameCOMMENT_BLOCK && token == TokenNameLBRACE;
523 // Do not add a newline & indent between a close brace and a colon
524 // (in array assignments, for example).
525 boolean commaAndCloseBrace = previousToken == TokenNameRBRACE && token == TokenNameCOMMA;
526 // Add a newline and indent, if appropriate.
528 || (!commentAndOpenBrace && !closeBraceAndCloseParen && !nlicsOption && !semiColonAndCloseBrace && !commaAndCloseBrace)) {
529 // if clearAllBlankLinesMode=false, leaves the blank lines
530 // inserted by the user
531 // if clearAllBlankLinesMode=true, removes all of then
532 // and insert only blank lines required by the formatting.
533 if (!options.clearAllBlankLinesMode) {
534 // (isComment(token))
535 pendingNewLines = (pendingNewLines < newLinesInWhitespace) ? newLinesInWhitespace : pendingNewLines;
536 pendingNewLines = (pendingNewLines > 2) ? 2 : pendingNewLines;
538 if (previousCompilableToken == TokenNameLBRACE && token == TokenNameRBRACE) {
539 containsOpenCloseBraces = true;
540 indentationLevelForOpenCloseBraces = currentLineIndentationLevel;
541 if (isComment(previousToken)) {
542 newLine(pendingNewLines);
545 * if (!(constructionsCount > 1 && constructions[constructionsCount-1] == NONINDENT_BLOCK &&
546 * (constructions[constructionsCount-2] == TokenNamefor
548 if (options.newLineInEmptyBlockMode) {
549 if (inArrayAssignment) {
550 newLine(1); // array assigment with an empty block
552 newLine(pendingNewLines);
558 // see PR 1FKKC3U: LFCOM:WINNT - Format problem with a comment
560 if (!((previousToken == Scanner.TokenNameCOMMENT_BLOCK || previousToken == Scanner.TokenNameCOMMENT_PHPDOC) && token == TokenNameSEMICOLON)) {
561 newLine(pendingNewLines);
564 if (((previousCompilableToken == TokenNameSEMICOLON) || (previousCompilableToken == TokenNameLBRACE)
565 || (previousCompilableToken == TokenNameRBRACE) || (isComment(previousToken)))
566 && (token == TokenNameRBRACE)) {
567 indentationOffset = -1;
568 indentationLevel += popExclusiveUntilBlock();
570 if (previousToken == Scanner.TokenNameCOMMENT_LINE && inAssignment) {
572 currentLineIndentationLevel++;
574 currentLineIndentationLevel = indentationLevel + indentationOffset;
576 pendingSpace = false;
577 indentationOffset = 0;
580 newLinesInWhitespace = 0;
582 if (nlicsToken == TokenNamedo && token == TokenNamewhile) {
586 boolean phpTagAndWhitespace = previousToken == TokenNameINLINE_HTML && token == TokenNameWHITESPACE;
588 // case TokenNameDOLLAR :
589 // dollarBraceCount++;
592 // case TokenNamefinally :
593 expectingOpenBrace = true;
594 pendingNewlineAfterParen = true;
595 indentationLevel += pushControlStatement(token);
598 case TokenNamedefault:
599 if (tokenBeforeColonCount == tokenBeforeColon.length) {
601 .arraycopy(tokenBeforeColon, 0, (tokenBeforeColon = new int[tokenBeforeColonCount * 2]), 0, tokenBeforeColonCount);
603 tokenBeforeColon[tokenBeforeColonCount++] = TokenNamecase;
604 indentationLevel += pushControlStatement(TokenNamecase);
606 case TokenNameQUESTION:
607 if (tokenBeforeColonCount == tokenBeforeColon.length) {
609 .arraycopy(tokenBeforeColon, 0, (tokenBeforeColon = new int[tokenBeforeColonCount * 2]), 0, tokenBeforeColonCount);
611 tokenBeforeColon[tokenBeforeColonCount++] = token;
613 case TokenNameswitch:
615 case TokenNameforeach:
618 if (openParenthesisCount == openParenthesis.length) {
619 System.arraycopy(openParenthesis, 0, (openParenthesis = new int[openParenthesisCount * 2]), 0, openParenthesisCount);
621 openParenthesis[openParenthesisCount++] = 0;
622 expectingOpenBrace = true;
623 indentationLevel += pushControlStatement(token);
626 pendingNewlineAfterParen = true;
628 // several CATCH statements can be contiguous.
629 // a CATCH is encountered pop until first CATCH (if a CATCH
630 // follows a TRY it works the same way,
631 // as CATCH and TRY are the same token in the stack).
632 expectingOpenBrace = true;
633 indentationLevel += pushControlStatement(TokenNamecatch);
636 expectingOpenBrace = true;
637 indentationLevel += pushControlStatement(token);
642 case TokenNameLPAREN:
643 // if (previousToken == TokenNamesynchronized) {
644 // indentationLevel += pushControlStatement(previousToken);
646 // Put a space between the previous and current token if the
647 // previous token was not a keyword, open paren, logical
648 // compliment (eg: !), semi-colon, open brace, close brace,
650 if (previousCompilableToken != TokenNameLBRACKET && previousToken != TokenNameIdentifier && previousToken != 0
651 && previousToken != TokenNameNOT && previousToken != TokenNameLPAREN && previousToken != TokenNameTWIDDLE
652 && previousToken != TokenNameSEMICOLON && previousToken != TokenNameLBRACE && previousToken != TokenNameRBRACE
653 && previousToken != TokenNamesuper) {
654 // && previousToken != TokenNamethis) {
657 // If in a for/if/while statement, increase the parenthesis count
658 // for the current openParenthesisCount
659 // else increase the count for stand alone parenthesis.
660 if (openParenthesisCount > 0)
661 openParenthesis[openParenthesisCount - 1]++;
663 openParenthesis[0]++;
664 pendingSpace = false;
667 case TokenNameRPAREN:
668 // Decrease the parenthesis count
669 // if there is no more unclosed parenthesis,
670 // a new line and indent may be append (depending on the next
672 if ((openParenthesisCount > 1) && (openParenthesis[openParenthesisCount - 1] > 0)) {
673 openParenthesis[openParenthesisCount - 1]--;
674 if (openParenthesis[openParenthesisCount - 1] <= 0) {
675 pendingNewlineAfterParen = true;
676 inAssignment = false;
677 openParenthesisCount--;
680 openParenthesis[0]--;
682 pendingSpace = false;
684 case TokenNameLBRACE:
685 if (previousCompilableToken == TokenNameDOLLAR) {
688 if ((previousCompilableToken == TokenNameRBRACKET) || (previousCompilableToken == TokenNameEQUAL)) {
689 // if (previousCompilableToken == TokenNameRBRACKET) {
690 inArrayAssignment = true;
691 inAssignment = false;
693 if (inArrayAssignment) {
694 indentationLevel += pushBlock();
696 // Add new line and increase indentation level after open brace.
698 indentationLevel += pushBlock();
702 case TokenNameRBRACE:
703 if (dollarBraceCount > 0) {
707 if (previousCompilableToken == TokenNameRPAREN) {
708 pendingSpace = false;
710 if (inArrayAssignment) {
711 inArrayAssignment = false;
713 indentationLevel += popInclusiveUntilBlock();
716 indentationLevel += popInclusiveUntilBlock();
717 if (previousCompilableToken == TokenNameRPAREN) {
718 // fix for 1FGDDV6: LFCOM:WIN98 - Weird splitting on message
720 currentLineBuffer.append(options.lineSeparatorSequence);
721 increaseLineDelta(options.lineSeparatorSequence.length);
723 if (constructionsCount > 0) {
724 switch (constructions[constructionsCount - 1]) {
726 case TokenNameforeach:
727 //indentationLevel += popExclusiveUntilBlock();
729 case TokenNameswitch:
734 case TokenNamefinally:
737 // case TokenNamesynchronized :
738 clearNonBlockIndents = true;
745 case TokenNameLBRACKET:
747 pendingSpace = false;
749 case TokenNameRBRACKET:
750 openBracketCount -= (openBracketCount > 0) ? 1 : 0;
751 // if there is no left bracket to close, the right bracket is
753 pendingSpace = false;
757 pendingSpace = false;
759 case TokenNameSEMICOLON:
760 // Do not generate line terminators in the definition of
761 // the for statement.
762 // if not in this case, jump a line and reduce indentation after
764 // if the block it closes belongs to a conditional statement (if,
766 if (openParenthesisCount <= 1) {
768 if (expectingOpenBrace) {
769 clearNonBlockIndents = true;
770 expectingOpenBrace = false;
773 inAssignment = false;
774 pendingSpace = false;
776 case TokenNamePLUS_PLUS:
777 case TokenNameMINUS_MINUS:
778 // Do not put a space between a post-increment/decrement
779 // and the identifier being modified.
780 if (previousToken == TokenNameIdentifier || previousToken == TokenNameRBRACKET) {
781 pendingSpace = false;
785 // previously ADDITION
787 // Handle the unary operators plus and minus via a flag
788 if (!isLiteralToken(previousToken) && previousToken != TokenNameIdentifier && previousToken != TokenNameRPAREN
789 && previousToken != TokenNameRBRACKET) {
790 unarySignModifier = 1;
794 // In a switch/case statement, add a newline & indent
795 // when a colon is encountered.
796 if (tokenBeforeColonCount > 0) {
797 if (tokenBeforeColon[tokenBeforeColonCount - 1] == TokenNamecase) {
800 tokenBeforeColonCount--;
806 case Scanner.TokenNameCOMMENT_LINE:
809 currentLineIndentationLevel++;
811 break; // a line is always inserted after a one-line comment
812 case Scanner.TokenNameCOMMENT_PHPDOC:
813 case Scanner.TokenNameCOMMENT_BLOCK:
814 currentCommentOffset = getCurrentCommentOffset();
817 case Scanner.TokenNameWHITESPACE:
818 if (!phpTagAndWhitespace) {
819 // Count the number of line terminators in the whitespace so
820 // line spacing can be preserved near comments.
821 char[] source = scanner.source;
822 newLinesInWhitespace = 0;
823 for (int i = scanner.startPosition, max = scanner.currentPosition; i < max; i++) {
824 if (source[i] == '\r') {
826 if (source[++i] == '\n') {
827 newLinesInWhitespace++;
829 newLinesInWhitespace++;
832 newLinesInWhitespace++;
834 } else if (source[i] == '\n') {
835 newLinesInWhitespace++;
838 increaseLineDelta(scanner.startPosition - scanner.currentPosition);
841 // case TokenNameHTML :
842 // // Add the next token to the formatted source string.
843 // // outputCurrentToken(token);
844 // int startPosition = scanner.startPosition;
846 // for (int i = startPosition, max = scanner.currentPosition; i <
848 // char currentCharacter = scanner.source[i];
849 // updateMappedPositions(i);
850 // currentLineBuffer.append(currentCharacter);
854 if ((token == TokenNameIdentifier) || isLiteralToken(token) || token == TokenNamesuper) {
855 // || token == TokenNamethis) {
856 // Do not put a space between a unary operator
857 // (eg: ++, --, +, -) and the identifier being modified.
858 if (previousToken == TokenNamePLUS_PLUS || previousToken == TokenNameMINUS_MINUS
859 || (previousToken == TokenNameMINUS_GREATER && options.compactDereferencingMode) // ->
860 || (previousToken == TokenNamePLUS && unarySignModifier > 0)
861 || (previousToken == TokenNameMINUS && unarySignModifier > 0)) {
862 pendingSpace = false;
864 unarySignModifier = 0;
868 // Do not output whitespace tokens.
869 if (token != Scanner.TokenNameWHITESPACE || phpTagAndWhitespace) {
871 * Add pending space to the formatted source string. Do not output a space under the following circumstances: 1) this is
872 * the first pass 2) previous token is an open paren 3) previous token is a period 4) previous token is the logical
873 * compliment (eg: !) 5) previous token is the bitwise compliment (eg: ~) 6) previous token is the open bracket (eg: [) 7)
874 * in an assignment statement, if the previous token is an open brace or the current token is a close brace 8) previous
875 * token is a single line comment 9) current token is a '->'
877 if (token == TokenNameMINUS_GREATER && options.compactDereferencingMode)
878 pendingSpace = false;
880 boolean openAndCloseBrace = previousCompilableToken == TokenNameLBRACE && token == TokenNameRBRACE;
881 if (pendingSpace && insertSpaceAfter(previousToken)
882 && !(inAssignment && (previousToken == TokenNameLBRACE || token == TokenNameRBRACE))
883 && previousToken != Scanner.TokenNameCOMMENT_LINE) {
884 if ((!(options.compactAssignmentMode && token == TokenNameEQUAL)) && !openAndCloseBrace)
887 // Add the next token to the formatted source string.
888 outputCurrentToken(token);
889 if (token == Scanner.TokenNameCOMMENT_LINE && openParenthesisCount > 1) {
891 currentLineBuffer.append(options.lineSeparatorSequence);
892 increaseLineDelta(options.lineSeparatorSequence.length);
896 // Whitespace tokens do not need to be remembered.
897 if (token != Scanner.TokenNameWHITESPACE || phpTagAndWhitespace) {
898 previousToken = token;
899 if (token != Scanner.TokenNameCOMMENT_BLOCK && token != Scanner.TokenNameCOMMENT_LINE
900 && token != Scanner.TokenNameCOMMENT_PHPDOC) {
901 previousCompilableToken = token;
905 output(copyRemainingSource());
907 // dump the last token of the source in the formatted output.
908 } catch (InvalidInputException e) {
909 output(copyRemainingSource());
911 // dump the last token of the source in the formatted output.
916 * Formats the char array <code>sourceString</code>, and returns a string containing the formatted version.
918 * @return the formatted ouput.
920 public String formatSourceString(String sourceString) {
921 char[] sourceChars = sourceString.toCharArray();
922 formattedSource = new StringBuffer(sourceChars.length);
923 scanner.setSource(sourceChars);
925 return formattedSource.toString();
929 * Formats the char array <code>sourceString</code>, and returns a string containing the formatted version.
932 * the string to format
933 * @param indentationLevel
934 * the initial indentation level
935 * @return the formatted ouput.
937 public String format(String string, int indentationLevel) {
938 return format(string, indentationLevel, (int[]) null);
942 * Formats the char array <code>sourceString</code>, and returns a string containing the formatted version. The positions array
943 * is modified to contain the mapped positions.
946 * the string to format
947 * @param indentationLevel
948 * the initial indentation level
950 * the array of positions to map
951 * @return the formatted ouput.
953 public String format(String string, int indentationLevel, int[] positions) {
954 return this.format(string, indentationLevel, positions, null);
957 public String format(String string, int indentationLevel, int[] positions, String lineSeparator) {
958 if (lineSeparator != null) {
959 this.options.setLineSeparator(lineSeparator);
961 if (positions != null) {
962 this.setPositionsToMap(positions);
963 this.setInitialIndentationLevel(indentationLevel);
964 String formattedString = this.formatSourceString(string);
965 int[] mappedPositions = this.getMappedPositions();
966 System.arraycopy(mappedPositions, 0, positions, 0, positions.length);
967 return formattedString;
969 this.setInitialIndentationLevel(indentationLevel);
970 return this.formatSourceString(string);
975 * Formats the char array <code>sourceString</code>, and returns a string containing the formatted version. The initial
976 * indentation level is 0.
979 * the string to format
980 * @return the formatted ouput.
982 public String format(String string) {
983 return this.format(string, 0, (int[]) null);
987 * Formats a given source string, starting indenting it at a particular depth and using the given options
989 * @deprecated backport 1.0 internal functionality
991 public static String format(String sourceString, int initialIndentationLevel, ConfigurableOption[] options) {
992 CodeFormatter formatter = new CodeFormatter(options);
993 formatter.setInitialIndentationLevel(initialIndentationLevel);
994 return formatter.formatSourceString(sourceString);
998 * Returns the number of characters and tab char between the beginning of the line and the beginning of the comment.
1000 private int getCurrentCommentOffset() {
1001 int linePtr = scanner.linePtr;
1002 // if there is no beginning of line, return 0.
1006 int beginningOfLine = scanner.lineEnds[linePtr];
1007 int currentStartPosition = scanner.startPosition;
1008 char[] source = scanner.source;
1009 // find the position of the beginning of the line containing the comment
1010 while (beginningOfLine > currentStartPosition) {
1012 beginningOfLine = scanner.lineEnds[--linePtr];
1014 beginningOfLine = 0;
1018 for (int i = currentStartPosition - 1; i >= beginningOfLine; i--) {
1019 char currentCharacter = source[i];
1020 switch (currentCharacter) {
1022 offset += options.tabSize;
1038 * Returns an array of descriptions for the configurable options. The descriptions may be changed and passed back to a different
1041 * @deprecated backport 1.0 internal functionality
1043 public static ConfigurableOption[] getDefaultOptions(Locale locale) {
1044 String componentName = CodeFormatter.class.getName();
1045 FormatterOptions options = new FormatterOptions();
1046 return new ConfigurableOption[] {
1047 new ConfigurableOption(componentName, "newline.openingBrace", locale, options.newLineBeforeOpeningBraceMode ? 0 : 1),
1049 new ConfigurableOption(componentName, "newline.controlStatement", locale, options.newlineInControlStatementMode ? 0 : 1),
1051 new ConfigurableOption(componentName, "newline.clearAll", locale, options.clearAllBlankLinesMode ? 0 : 1),
1053 // new ConfigurableOption(componentName, "newline.elseIf", locale,
1054 // options.compactElseIfMode ? 0 : 1), //$NON-NLS-1$
1055 new ConfigurableOption(componentName, "newline.emptyBlock", locale, options.newLineInEmptyBlockMode ? 0 : 1),
1057 new ConfigurableOption(componentName, "line.split", locale, options.maxLineLength),
1059 new ConfigurableOption(componentName, "style.compactAssignment", locale, options.compactAssignmentMode ? 0 : 1),
1061 new ConfigurableOption(componentName, "tabulation.char", locale, options.indentWithTab ? 0 : 1),
1063 new ConfigurableOption(componentName, "tabulation.size", locale, options.tabSize) //$NON-NLS-1$
1068 * Returns the array of mapped positions. Returns null is no positions have been set.
1071 * @deprecated There is no need to retrieve the mapped positions anymore.
1073 public int[] getMappedPositions() {
1074 return mappedPositions;
1078 * Returns the priority of the token given as argument <br>
1079 * The most prioritary the token is, the smallest the return value is.
1081 * @return the priority of <code>token</code>
1083 * the token of which the priority is requested
1085 private static int getTokenPriority(int token) {
1087 case TokenNameextends:
1088 // case TokenNameimplements :
1089 // case TokenNamethrows :
1091 case TokenNameSEMICOLON:
1094 case TokenNameCOMMA:
1097 case TokenNameEQUAL:
1100 case TokenNameAND_AND:
1102 case TokenNameOR_OR:
1105 case TokenNameQUESTION:
1107 case TokenNameCOLON:
1109 return 50; // it's better cutting on ?: than on ;
1110 case TokenNameEQUAL_EQUAL:
1112 case TokenNameEQUAL_EQUAL_EQUAL:
1114 case TokenNameNOT_EQUAL:
1116 case TokenNameNOT_EQUAL_EQUAL:
1121 case TokenNameLESS_EQUAL:
1123 case TokenNameGREATER:
1125 case TokenNameGREATER_EQUAL:
1127 // case TokenNameinstanceof : // instanceof
1131 case TokenNameMINUS:
1134 case TokenNameMULTIPLY:
1136 case TokenNameDIVIDE:
1138 case TokenNameREMAINDER:
1141 case TokenNameLEFT_SHIFT:
1143 case TokenNameRIGHT_SHIFT:
1145 // case TokenNameUNSIGNED_RIGHT_SHIFT : // >>>
1154 case TokenNameMULTIPLY_EQUAL:
1156 case TokenNameDIVIDE_EQUAL:
1158 case TokenNameREMAINDER_EQUAL:
1160 case TokenNamePLUS_EQUAL:
1162 case TokenNameMINUS_EQUAL:
1164 case TokenNameLEFT_SHIFT_EQUAL:
1166 case TokenNameRIGHT_SHIFT_EQUAL:
1168 // case TokenNameUNSIGNED_RIGHT_SHIFT_EQUAL : // >>>=
1169 case TokenNameAND_EQUAL:
1171 case TokenNameXOR_EQUAL:
1173 case TokenNameOR_EQUAL:
1175 case TokenNameDOT_EQUAL:
1182 return Integer.MAX_VALUE;
1187 * Handles the exception raised when an invalid token is encountered. Returns true if the exception has been handled, false
1190 private boolean handleInvalidToken(Exception e) {
1191 if (e.getMessage().equals(Scanner.INVALID_CHARACTER_CONSTANT) || e.getMessage().equals(Scanner.INVALID_CHAR_IN_STRING)
1192 || e.getMessage().equals(Scanner.INVALID_ESCAPE)) {
1198 private final void increaseGlobalDelta(int offset) {
1199 globalDelta += offset;
1202 private final void increaseLineDelta(int offset) {
1203 lineDelta += offset;
1206 private final void increaseSplitDelta(int offset) {
1207 splitDelta += offset;
1211 * Returns true if a space has to be inserted after <code>operator</code> false otherwise.
1213 private boolean insertSpaceAfter(int token) {
1215 case TokenNameLPAREN:
1217 case TokenNameTWIDDLE:
1221 case TokenNameWHITESPACE:
1222 case TokenNameLBRACKET:
1223 case TokenNameDOLLAR:
1224 case Scanner.TokenNameCOMMENT_LINE:
1232 * Returns true if a space has to be inserted before <code>operator</code> false otherwise. <br>
1233 * Cannot be static as it uses the code formatter options (to know if the compact assignment mode is on).
1235 private boolean insertSpaceBefore(int token) {
1237 case TokenNameEQUAL:
1238 return (!options.compactAssignmentMode);
1244 private static boolean isComment(int token) {
1245 boolean result = token == Scanner.TokenNameCOMMENT_BLOCK || token == Scanner.TokenNameCOMMENT_LINE
1246 || token == Scanner.TokenNameCOMMENT_PHPDOC;
1250 private static boolean isLiteralToken(int token) {
1251 boolean result = token == TokenNameIntegerLiteral
1252 // || token == TokenNameLongLiteral
1253 // || token == TokenNameFloatingPointLiteral
1254 || token == TokenNameDoubleLiteral
1255 // || token == TokenNameCharacterLiteral
1256 || token == TokenNameStringDoubleQuote;
1261 * If the length of <code>oneLineBuffer</code> exceeds <code>maxLineLength</code>, it is split and the result is dumped in
1262 * <code>formattedSource</code>
1264 * @param newLineCount
1265 * the number of new lines to append
1267 private void newLine(int newLineCount) {
1268 // format current line
1270 beginningOfLineIndex = formattedSource.length();
1271 String currentLine = currentLineBuffer.toString();
1272 if (containsOpenCloseBraces) {
1273 containsOpenCloseBraces = false;
1274 outputLine(currentLine, false, indentationLevelForOpenCloseBraces, 0, -1, null, 0);
1275 indentationLevelForOpenCloseBraces = currentLineIndentationLevel;
1277 outputLine(currentLine, false, currentLineIndentationLevel, 0, -1, null, 0);
1279 // dump line break(s)
1280 for (int i = 0; i < newLineCount; i++) {
1281 formattedSource.append(options.lineSeparatorSequence);
1282 increaseSplitDelta(options.lineSeparatorSequence.length);
1284 // reset formatter for next line
1285 int currentLength = currentLine.length();
1286 currentLineBuffer = new StringBuffer(currentLength > maxLineSize ? maxLineSize = currentLength : maxLineSize);
1287 increaseGlobalDelta(splitDelta);
1288 increaseGlobalDelta(lineDelta);
1290 currentLineIndentationLevel = initialIndentationLevel;
1293 private String operatorString(int operator) {
1295 case TokenNameextends:
1296 return "extends"; //$NON-NLS-1$
1297 // case TokenNameimplements :
1298 // return "implements"; //$NON-NLS-1$
1300 // case TokenNamethrows :
1301 // return "throws"; //$NON-NLS-1$
1302 case TokenNameSEMICOLON:
1304 return ";"; //$NON-NLS-1$
1305 case TokenNameCOMMA:
1307 return ","; //$NON-NLS-1$
1308 case TokenNameEQUAL:
1310 return "="; //$NON-NLS-1$
1311 case TokenNameAND_AND:
1313 return "&&"; //$NON-NLS-1$
1314 case TokenNameOR_OR:
1316 return "||"; //$NON-NLS-1$
1317 case TokenNameQUESTION:
1319 return "?"; //$NON-NLS-1$
1320 case TokenNameCOLON:
1322 return ":"; //$NON-NLS-1$
1323 case TokenNamePAAMAYIM_NEKUDOTAYIM:
1325 return "::"; //$NON-NLS-1$
1326 case TokenNameEQUAL_EQUAL:
1327 // == (15.20, 15.20.1, 15.20.2, 15.20.3)
1328 return "=="; //$NON-NLS-1$
1329 case TokenNameEQUAL_EQUAL_EQUAL:
1330 // == (15.20, 15.20.1, 15.20.2, 15.20.3)
1331 return "==="; //$NON-NLS-1$
1332 case TokenNameEQUAL_GREATER:
1334 return "=>"; //$NON-NLS-1$
1335 case TokenNameNOT_EQUAL:
1336 // != (15.20, 15.20.1, 15.20.2, 15.20.3)
1337 return "!="; //$NON-NLS-1$
1338 case TokenNameNOT_EQUAL_EQUAL:
1339 // != (15.20, 15.20.1, 15.20.2, 15.20.3)
1340 return "!=="; //$NON-NLS-1$
1343 return "<"; //$NON-NLS-1$
1344 case TokenNameLESS_EQUAL:
1346 return "<="; //$NON-NLS-1$
1347 case TokenNameGREATER:
1349 return ">"; //$NON-NLS-1$
1350 case TokenNameGREATER_EQUAL:
1352 return ">="; //$NON-NLS-1$
1353 // case TokenNameinstanceof : // instanceof
1354 // return "instanceof"; //$NON-NLS-1$
1356 // + (15.17, 15.17.2)
1357 return "+"; //$NON-NLS-1$
1358 case TokenNameMINUS:
1360 return "-"; //$NON-NLS-1$
1361 case TokenNameMULTIPLY:
1363 return "*"; //$NON-NLS-1$
1364 case TokenNameDIVIDE:
1366 return "/"; //$NON-NLS-1$
1367 case TokenNameREMAINDER:
1369 return "%"; //$NON-NLS-1$
1370 case TokenNameLEFT_SHIFT:
1372 return "<<"; //$NON-NLS-1$
1373 case TokenNameRIGHT_SHIFT:
1375 return ">>"; //$NON-NLS-1$
1376 // case TokenNameUNSIGNED_RIGHT_SHIFT : // >>> (15.18)
1377 // return ">>>"; //$NON-NLS-1$
1379 // & (15.21, 15.21.1, 15.21.2)
1380 return "&"; //$NON-NLS-1$
1382 // | (15.21, 15.21.1, 15.21.2)
1383 return "|"; //$NON-NLS-1$
1385 // ^ (15.21, 15.21.1, 15.21.2)
1386 return "^"; //$NON-NLS-1$
1387 case TokenNameMULTIPLY_EQUAL:
1389 return "*="; //$NON-NLS-1$
1390 case TokenNameDIVIDE_EQUAL:
1392 return "/="; //$NON-NLS-1$
1393 case TokenNameREMAINDER_EQUAL:
1395 return "%="; //$NON-NLS-1$
1396 case TokenNamePLUS_EQUAL:
1398 return "+="; //$NON-NLS-1$
1399 case TokenNameMINUS_EQUAL:
1401 return "-="; //$NON-NLS-1$
1402 case TokenNameMINUS_GREATER:
1404 return "->"; //$NON-NLS-1$
1405 case TokenNameLEFT_SHIFT_EQUAL:
1407 return "<<="; //$NON-NLS-1$
1408 case TokenNameRIGHT_SHIFT_EQUAL:
1410 return ">>="; //$NON-NLS-1$
1411 // case TokenNameUNSIGNED_RIGHT_SHIFT_EQUAL : // >>>= (15.25.2)
1412 // return ">>>="; //$NON-NLS-1$
1413 case TokenNameAND_EQUAL:
1415 return "&="; //$NON-NLS-1$
1416 case TokenNameXOR_EQUAL:
1418 return "^="; //$NON-NLS-1$
1419 case TokenNameOR_EQUAL:
1421 return "|="; //$NON-NLS-1$
1422 case TokenNameDOT_EQUAL:
1424 return ".="; //$NON-NLS-1$
1427 return "."; //$NON-NLS-1$
1429 return ""; //$NON-NLS-1$
1434 * Appends <code>stringToOutput</code> to the formatted output. <br>
1435 * If it contains \n, append a LINE_SEPARATOR and indent after it.
1437 private void output(String stringToOutput) {
1438 char currentCharacter;
1439 for (int i = 0, max = stringToOutput.length(); i < max; i++) {
1440 currentCharacter = stringToOutput.charAt(i);
1441 if (currentCharacter != '\t') {
1442 currentLineBuffer.append(currentCharacter);
1448 * Appends <code>token</code> to the formatted output. <br>
1449 * If it contains <code>\n</code>, append a LINE_SEPARATOR and indent after it.
1451 private void outputCurrentToken(int token) {
1452 char[] source = scanner.source;
1453 int startPosition = scanner.startPosition;
1455 case Scanner.TokenNameCOMMENT_PHPDOC:
1456 case Scanner.TokenNameCOMMENT_BLOCK:
1457 case Scanner.TokenNameCOMMENT_LINE:
1458 boolean endOfLine = false;
1459 int currentCommentOffset = getCurrentCommentOffset();
1460 int beginningOfLineSpaces = 0;
1462 currentCommentOffset = getCurrentCommentOffset();
1463 beginningOfLineSpaces = 0;
1464 boolean pendingCarriageReturn = false;
1465 for (int i = startPosition, max = scanner.currentPosition; i < max; i++) {
1466 char currentCharacter = source[i];
1467 updateMappedPositions(i);
1468 switch (currentCharacter) {
1470 pendingCarriageReturn = true;
1474 if (pendingCarriageReturn) {
1475 increaseGlobalDelta(options.lineSeparatorSequence.length - 2);
1477 increaseGlobalDelta(options.lineSeparatorSequence.length - 1);
1479 pendingCarriageReturn = false;
1480 currentLineBuffer.append(options.lineSeparatorSequence);
1481 beginningOfLineSpaces = 0;
1485 if (pendingCarriageReturn) {
1486 pendingCarriageReturn = false;
1487 increaseGlobalDelta(options.lineSeparatorSequence.length - 1);
1488 currentLineBuffer.append(options.lineSeparatorSequence);
1489 beginningOfLineSpaces = 0;
1493 // we remove a maximum of currentCommentOffset characters (tabs
1494 // are converted to space numbers).
1495 beginningOfLineSpaces += options.tabSize;
1496 if (beginningOfLineSpaces > currentCommentOffset) {
1497 currentLineBuffer.append(currentCharacter);
1499 increaseGlobalDelta(-1);
1502 currentLineBuffer.append(currentCharacter);
1506 if (pendingCarriageReturn) {
1507 pendingCarriageReturn = false;
1508 increaseGlobalDelta(options.lineSeparatorSequence.length - 1);
1509 currentLineBuffer.append(options.lineSeparatorSequence);
1510 beginningOfLineSpaces = 0;
1514 // we remove a maximum of currentCommentOffset characters (tabs
1515 // are converted to space numbers).
1516 beginningOfLineSpaces++;
1517 if (beginningOfLineSpaces > currentCommentOffset) {
1518 currentLineBuffer.append(currentCharacter);
1520 increaseGlobalDelta(-1);
1523 currentLineBuffer.append(currentCharacter);
1527 if (pendingCarriageReturn) {
1528 pendingCarriageReturn = false;
1529 increaseGlobalDelta(options.lineSeparatorSequence.length - 1);
1530 currentLineBuffer.append(options.lineSeparatorSequence);
1531 beginningOfLineSpaces = 0;
1534 beginningOfLineSpaces = 0;
1535 currentLineBuffer.append(currentCharacter);
1540 updateMappedPositions(scanner.currentPosition - 1);
1541 multipleLineCommentCounter++;
1544 for (int i = startPosition, max = scanner.currentPosition; i < max; i++) {
1545 char currentCharacter = source[i];
1546 updateMappedPositions(i);
1547 currentLineBuffer.append(currentCharacter);
1553 * Outputs <code>currentString</code>:<br>
1555 * <li>If its length is < maxLineLength, output
1556 * <li>Otherwise it is split.
1559 * @param currentString
1561 * @param preIndented
1562 * whether the string to output was pre-indented
1564 * number of indentation to put in front of <code>currentString</code>
1566 * value of the operator belonging to <code>currentString</code>.
1568 private void outputLine(String currentString, boolean preIndented, int depth, int operator, int substringIndex,
1569 int[] startSubstringIndexes, int offsetInGlobalLine) {
1570 boolean emptyFirstSubString = false;
1571 String operatorString = operatorString(operator);
1572 boolean placeOperatorBehind = !breakLineBeforeOperator(operator);
1573 boolean placeOperatorAhead = !placeOperatorBehind;
1574 // dump prefix operator?
1575 if (placeOperatorAhead) {
1580 if (operator != 0) {
1581 if (insertSpaceBefore(operator)) {
1582 formattedSource.append(' ');
1583 increaseSplitDelta(1);
1585 formattedSource.append(operatorString);
1586 increaseSplitDelta(operatorString.length());
1587 if (insertSpaceAfter(operator) && operator != TokenNameimplements && operator != TokenNameextends) {
1588 // && operator != TokenNamethrows) {
1589 formattedSource.append(' ');
1590 increaseSplitDelta(1);
1594 SplitLine splitLine = null;
1595 if (options.maxLineLength == 0 || getLength(currentString, depth) < options.maxLineLength
1596 || (splitLine = split(currentString, offsetInGlobalLine)) == null) {
1597 // depending on the type of operator, outputs new line before of after
1599 // indent before postfix operator
1600 // indent also when the line cannot be split
1601 if (operator == TokenNameextends || operator == TokenNameimplements) {
1602 // || operator == TokenNamethrows) {
1603 formattedSource.append(' ');
1604 increaseSplitDelta(1);
1606 if (placeOperatorBehind) {
1611 int max = currentString.length();
1612 if (multipleLineCommentCounter != 0) {
1614 BufferedReader reader = new BufferedReader(new StringReader(currentString));
1615 String line = reader.readLine();
1616 while (line != null) {
1617 updateMappedPositionsWhileSplitting(beginningOfLineIndex, beginningOfLineIndex + line.length()
1618 + options.lineSeparatorSequence.length);
1619 formattedSource.append(line);
1620 beginningOfLineIndex = beginningOfLineIndex + line.length();
1621 if ((line = reader.readLine()) != null) {
1622 formattedSource.append(options.lineSeparatorSequence);
1623 beginningOfLineIndex += options.lineSeparatorSequence.length;
1624 dumpTab(currentLineIndentationLevel);
1628 } catch (IOException e) {
1629 e.printStackTrace();
1632 updateMappedPositionsWhileSplitting(beginningOfLineIndex, beginningOfLineIndex + max);
1633 for (int i = 0; i < max; i++) {
1634 char currentChar = currentString.charAt(i);
1635 switch (currentChar) {
1640 // fix for 1FFYL5C: LFCOM:ALL - Incorrect indentation when
1641 // split with a comment inside a condition
1642 // a substring cannot end with a lineSeparatorSequence,
1643 // except if it has been added by format() after a one-line
1645 formattedSource.append(options.lineSeparatorSequence);
1646 // 1FGDDV6: LFCOM:WIN98 - Weird splitting on message expression
1651 formattedSource.append(currentChar);
1655 // update positions inside the mappedPositions table
1656 if (substringIndex != -1) {
1657 if (multipleLineCommentCounter == 0) {
1658 int startPosition = beginningOfLineIndex + startSubstringIndexes[substringIndex];
1659 updateMappedPositionsWhileSplitting(startPosition, startPosition + max);
1661 // compute the splitDelta resulting with the operator and blank removal
1662 if (substringIndex + 1 != startSubstringIndexes.length) {
1663 increaseSplitDelta(startSubstringIndexes[substringIndex] + max - startSubstringIndexes[substringIndex + 1]);
1666 // dump postfix operator?
1667 if (placeOperatorBehind) {
1668 if (insertSpaceBefore(operator)) {
1669 formattedSource.append(' ');
1670 if (operator != 0) {
1671 increaseSplitDelta(1);
1674 formattedSource.append(operatorString);
1675 if (operator != 0) {
1676 increaseSplitDelta(operatorString.length());
1681 // fix for 1FG0BA3: LFCOM:WIN98 - Weird splitting on interfaces
1682 // extends has to stand alone on a line when currentString has been split.
1683 if (options.maxLineLength != 0 && splitLine != null && (operator == TokenNameextends)) {
1684 // || operator == TokenNameimplements
1685 // || operator == TokenNamethrows)) {
1686 formattedSource.append(options.lineSeparatorSequence);
1687 increaseSplitDelta(options.lineSeparatorSequence.length);
1690 if (operator == TokenNameextends) {
1691 // || operator == TokenNameimplements
1692 // || operator == TokenNamethrows) {
1693 formattedSource.append(' ');
1694 increaseSplitDelta(1);
1697 // perform actual splitting
1698 String result[] = splitLine.substrings;
1699 int[] splitOperators = splitLine.operators;
1700 if (result[0].length() == 0) {
1701 // when the substring 0 is null, the substring 1 is correctly indented.
1703 emptyFirstSubString = true;
1705 // the operator going in front of the result[0] string is the operator
1707 for (int i = 0, max = result.length; i < max; i++) {
1708 // the new depth is the current one if this is the first substring,
1709 // the current one + 1 otherwise.
1710 // if the substring is a comment, use the current indentation Level
1711 // instead of the depth
1712 // (-1 because the ouputline increases depth).
1713 // (fix for 1FFC72R: LFCOM:ALL - Incorrect line split in presence of line
1715 String currentResult = result[i];
1716 if (currentResult.length() != 0 || splitOperators[i] != 0) {
1717 int newDepth = (currentResult.startsWith("/*") //$NON-NLS-1$
1718 || currentResult.startsWith("//")) //$NON-NLS-1$
1719 ? indentationLevel - 1 : depth;
1720 outputLine(currentResult, i == 0 || (i == 1 && emptyFirstSubString) ? preIndented : false,
1721 i == 0 ? newDepth : newDepth + 1, splitOperators[i], i, splitLine.startSubstringsIndexes, currentString
1722 .indexOf(currentResult));
1724 formattedSource.append(options.lineSeparatorSequence);
1725 increaseSplitDelta(options.lineSeparatorSequence.length);
1729 if (result.length == splitOperators.length - 1) {
1730 int lastOperator = splitOperators[result.length];
1731 String lastOperatorString = operatorString(lastOperator);
1732 formattedSource.append(options.lineSeparatorSequence);
1733 increaseSplitDelta(options.lineSeparatorSequence.length);
1734 if (breakLineBeforeOperator(lastOperator)) {
1736 if (lastOperator != 0) {
1737 if (insertSpaceBefore(lastOperator)) {
1738 formattedSource.append(' ');
1739 increaseSplitDelta(1);
1741 formattedSource.append(lastOperatorString);
1742 increaseSplitDelta(lastOperatorString.length());
1743 if (insertSpaceAfter(lastOperator) && lastOperator != TokenNameimplements && lastOperator != TokenNameextends) {
1744 // && lastOperator != TokenNamethrows) {
1745 formattedSource.append(' ');
1746 increaseSplitDelta(1);
1751 if (placeOperatorBehind) {
1752 if (insertSpaceBefore(operator)) {
1753 formattedSource.append(' ');
1754 increaseSplitDelta(1);
1756 formattedSource.append(operatorString);
1757 //increaseSplitDelta(operatorString.length());
1762 * Pops the top statement of the stack if it is <code>token</code>
1764 private int pop(int token) {
1766 if ((constructionsCount > 0) && (constructions[constructionsCount - 1] == token)) {
1768 constructionsCount--;
1774 * Pops the top statement of the stack if it is a <code>BLOCK</code> or a <code>NONINDENT_BLOCK</code>.
1776 private int popBlock() {
1778 if ((constructionsCount > 0)
1779 && ((constructions[constructionsCount - 1] == BLOCK) || (constructions[constructionsCount - 1] == NONINDENT_BLOCK))) {
1780 if (constructions[constructionsCount - 1] == BLOCK)
1782 constructionsCount--;
1788 * Pops elements until the stack is empty or the top element is <code>token</code>.<br>
1789 * Does not remove <code>token</code> from the stack.
1792 * the token to be left as the top of the stack
1794 private int popExclusiveUntil(int token) {
1796 int startCount = constructionsCount;
1797 for (int i = startCount - 1; i >= 0 && constructions[i] != token; i--) {
1798 if (constructions[i] != NONINDENT_BLOCK)
1800 constructionsCount--;
1806 * Pops elements until the stack is empty or the top element is a <code>BLOCK</code> or a <code>NONINDENT_BLOCK</code>.<br>
1807 * Does not remove it from the stack.
1809 private int popExclusiveUntilBlock() {
1810 int startCount = constructionsCount;
1812 for (int i = startCount - 1; i >= 0 && constructions[i] != BLOCK && constructions[i] != NONINDENT_BLOCK; i--) {
1813 constructionsCount--;
1820 * Pops elements until the stack is empty or the top element is a <code>BLOCK</code>, a <code>NONINDENT_BLOCK</code> or a
1821 * <code>CASE</code>.<br>
1822 * Does not remove it from the stack.
1824 private int popExclusiveUntilBlockOrCase() {
1825 int startCount = constructionsCount;
1827 for (int i = startCount - 1; i >= 0 && constructions[i] != BLOCK && constructions[i] != NONINDENT_BLOCK
1828 && constructions[i] != TokenNamecase; i--) {
1829 constructionsCount--;
1836 * Pops elements until the stack is empty or the top element is <code>token</code>.<br>
1837 * Removes <code>token</code> from the stack too.
1840 * the token to remove from the stack
1842 private int popInclusiveUntil(int token) {
1843 int startCount = constructionsCount;
1845 for (int i = startCount - 1; i >= 0 && constructions[i] != token; i--) {
1846 if (constructions[i] != NONINDENT_BLOCK)
1848 constructionsCount--;
1850 if (constructionsCount > 0) {
1851 if (constructions[constructionsCount - 1] != NONINDENT_BLOCK)
1853 constructionsCount--;
1859 * Pops elements until the stack is empty or the top element is a <code>BLOCK</code> or a <code>NONINDENT_BLOCK</code>.<br>
1860 * Does not remove it from the stack.
1862 private int popInclusiveUntilBlock() {
1863 int startCount = constructionsCount;
1865 for (int i = startCount - 1; i >= 0 && (constructions[i] != BLOCK && constructions[i] != NONINDENT_BLOCK); i--) {
1867 constructionsCount--;
1869 if (constructionsCount > 0) {
1870 if (constructions[constructionsCount - 1] == BLOCK)
1872 constructionsCount--;
1878 * Pushes a block in the stack. <br>
1879 * Pushes a <code>BLOCK</code> if the stack is empty or if the top element is a <code>BLOCK</code>, pushes
1880 * <code>NONINDENT_BLOCK</code> otherwise. Creates a new bigger array if the current one is full.
1882 private int pushBlock() {
1884 if (constructionsCount == constructions.length)
1885 System.arraycopy(constructions, 0, (constructions = new int[constructionsCount * 2]), 0, constructionsCount);
1886 if ((constructionsCount == 0) || (constructions[constructionsCount - 1] == BLOCK)
1887 || (constructions[constructionsCount - 1] == NONINDENT_BLOCK) || (constructions[constructionsCount - 1] == TokenNamecase)) {
1889 constructions[constructionsCount++] = BLOCK;
1891 constructions[constructionsCount++] = NONINDENT_BLOCK;
1897 * Pushes <code>token</code>.<br>
1898 * Creates a new bigger array if the current one is full.
1900 private int pushControlStatement(int token) {
1901 if (constructionsCount == constructions.length)
1902 System.arraycopy(constructions, 0, (constructions = new int[constructionsCount * 2]), 0, constructionsCount);
1903 constructions[constructionsCount++] = token;
1907 private static boolean separateFirstArgumentOn(int currentToken) {
1908 //return (currentToken == TokenNameCOMMA || currentToken ==
1909 // TokenNameSEMICOLON);
1910 return currentToken != TokenNameif && currentToken != TokenNameLPAREN && currentToken != TokenNameNOT
1911 && currentToken != TokenNamewhile && currentToken != TokenNamefor && currentToken != TokenNameforeach
1912 && currentToken != TokenNameswitch;
1916 * Set the positions to map. The mapped positions should be retrieved using the getMappedPositions() method.
1920 * @deprecated Set the positions to map using the format(String, int, int[]) method.
1922 * @see #getMappedPositions()
1924 public void setPositionsToMap(int[] positions) {
1925 positionsToMap = positions;
1928 mappedPositions = new int[positions.length];
1932 * Appends a space character to the current line buffer.
1934 private void space() {
1935 currentLineBuffer.append(' ');
1936 increaseLineDelta(1);
1940 * Splits <code>stringToSplit</code> on the top level token <br>
1941 * If there are several identical token at the same level, the string is cut into many pieces.
1943 * @return an object containing the operator and all the substrings or null if the string cannot be split
1945 public SplitLine split(String stringToSplit) {
1946 return split(stringToSplit, 0);
1950 * Splits <code>stringToSplit</code> on the top level token <br>
1951 * If there are several identical token at the same level, the string is cut into many pieces.
1953 * @return an object containing the operator and all the substrings or null if the string cannot be split
1955 public SplitLine split(String stringToSplit, int offsetInGlobalLine) {
1957 * See http://dev.eclipse.org/bugs/show_bug.cgi?id=12540 and http://dev.eclipse.org/bugs/show_bug.cgi?id=14387
1959 if (stringToSplit.indexOf("//$NON-NLS") != -1) { //$NON-NLS-1$
1962 // split doesn't work correct for PHP
1965 // int currentToken = 0;
1966 // int splitTokenType = 0;
1967 // int splitTokenDepth = Integer.MAX_VALUE;
1968 // int splitTokenPriority = Integer.MAX_VALUE;
1969 // int[] substringsStartPositions = new int[10];
1970 // // contains the start position of substrings
1971 // int[] substringsEndPositions = new int[10];
1972 // // contains the start position of substrings
1973 // int substringsCount = 1; // index in the substringsStartPosition array
1974 // int[] splitOperators = new int[10];
1975 // // contains the start position of substrings
1976 // int splitOperatorsCount = 0; // index in the substringsStartPosition array
1977 // int[] openParenthesisPosition = new int[10];
1978 // int openParenthesisPositionCount = 0;
1979 // int position = 0;
1980 // int lastOpenParenthesisPosition = -1;
1981 // // used to remember the position of the 1st open parenthesis
1982 // // needed for a pattern like: A.B(C); we want formatted like A.B( split C);
1983 // // setup the scanner with a new source
1984 // int lastCommentStartPosition = -1;
1985 // // to remember the start position of the last comment
1986 // int firstTokenOnLine = -1;
1987 // // to remember the first token of the line
1988 // int previousToken = -1;
1989 // // to remember the previous token.
1990 // splitScanner.setSource(stringToSplit.toCharArray());
1992 // // start the loop
1994 // // takes the next token
1996 // if (currentToken != Scanner.TokenNameWHITESPACE)
1997 // previousToken = currentToken;
1998 // currentToken = splitScanner.getNextToken();
1999 // if (Scanner.DEBUG) {
2000 // int currentEndPosition = splitScanner.getCurrentTokenEndPosition();
2001 // int currentStartPosition = splitScanner
2002 // .getCurrentTokenStartPosition();
2003 // System.out.print(currentStartPosition + "," + currentEndPosition
2005 // System.out.println(scanner.toStringAction(currentToken));
2007 // } catch (InvalidInputException e) {
2008 // if (!handleInvalidToken(e))
2010 // currentToken = 0;
2011 // // this value is not modify when an exception is raised.
2013 // if (currentToken == TokenNameEOF)
2015 // if (firstTokenOnLine == -1) {
2016 // firstTokenOnLine = currentToken;
2018 // switch (currentToken) {
2019 // case TokenNameRBRACE :
2020 // case TokenNameRPAREN :
2021 // if (openParenthesisPositionCount > 0) {
2022 // if (openParenthesisPositionCount == 1
2023 // && lastOpenParenthesisPosition < openParenthesisPosition[0]) {
2024 // lastOpenParenthesisPosition = openParenthesisPosition[0];
2025 // } else if ((splitTokenDepth == Integer.MAX_VALUE)
2026 // || (splitTokenDepth > openParenthesisPositionCount && openParenthesisPositionCount == 1)) {
2027 // splitTokenType = 0;
2028 // splitTokenDepth = openParenthesisPositionCount;
2029 // splitTokenPriority = Integer.MAX_VALUE;
2030 // substringsStartPositions[0] = 0;
2031 // // better token means the whole line until now is the first
2033 // substringsCount = 1; // resets the count of substrings
2034 // substringsEndPositions[0] = openParenthesisPosition[0];
2035 // // substring ends on operator start
2036 // position = openParenthesisPosition[0];
2037 // // the string mustn't be cut before the closing parenthesis but
2038 // // after the opening one.
2039 // splitOperatorsCount = 1; // resets the count of split operators
2040 // splitOperators[0] = 0;
2042 // openParenthesisPositionCount--;
2045 // case TokenNameLBRACE :
2046 // case TokenNameLPAREN :
2047 // if (openParenthesisPositionCount == openParenthesisPosition.length) {
2050 // openParenthesisPosition,
2052 // (openParenthesisPosition = new int[openParenthesisPositionCount * 2]),
2053 // 0, openParenthesisPositionCount);
2055 // openParenthesisPosition[openParenthesisPositionCount++] = splitScanner.currentPosition;
2056 // if (currentToken == TokenNameLPAREN
2057 // && previousToken == TokenNameRPAREN) {
2058 // openParenthesisPosition[openParenthesisPositionCount - 1] = splitScanner.startPosition;
2061 // case TokenNameSEMICOLON :
2063 // case TokenNameCOMMA :
2065 // case TokenNameEQUAL :
2067 // if (openParenthesisPositionCount < splitTokenDepth
2068 // || (openParenthesisPositionCount == splitTokenDepth && splitTokenPriority > getTokenPriority(currentToken))) {
2069 // // the current token is better than the one we currently have
2070 // // (in level or in priority if same level)
2071 // // reset the substringsCount
2072 // splitTokenDepth = openParenthesisPositionCount;
2073 // splitTokenType = currentToken;
2074 // splitTokenPriority = getTokenPriority(currentToken);
2075 // substringsStartPositions[0] = 0;
2076 // // better token means the whole line until now is the first
2078 // if (separateFirstArgumentOn(firstTokenOnLine)
2079 // && openParenthesisPositionCount > 0) {
2080 // substringsCount = 2; // resets the count of substrings
2081 // substringsEndPositions[0] = openParenthesisPosition[splitTokenDepth - 1];
2082 // substringsStartPositions[1] = openParenthesisPosition[splitTokenDepth - 1];
2083 // substringsEndPositions[1] = splitScanner.startPosition;
2084 // splitOperatorsCount = 2; // resets the count of split operators
2085 // splitOperators[0] = 0;
2086 // splitOperators[1] = currentToken;
2087 // position = splitScanner.currentPosition;
2088 // // next substring will start from operator end
2090 // substringsCount = 1; // resets the count of substrings
2091 // substringsEndPositions[0] = splitScanner.startPosition;
2092 // // substring ends on operator start
2093 // position = splitScanner.currentPosition;
2094 // // next substring will start from operator end
2095 // splitOperatorsCount = 1; // resets the count of split operators
2096 // splitOperators[0] = currentToken;
2099 // if ((openParenthesisPositionCount == splitTokenDepth && splitTokenPriority == getTokenPriority(currentToken))
2100 // && splitTokenType != TokenNameEQUAL
2101 // && currentToken != TokenNameEQUAL) {
2102 // // fix for 1FG0BCN: LFCOM:WIN98 - Missing one indentation after
2104 // // take only the 1st = into account.
2105 // // if another token with the same priority is found,
2106 // // push the start position of the substring and
2107 // // push the token into the stack.
2108 // // create a new array object if the current one is full.
2109 // if (substringsCount == substringsStartPositions.length) {
2112 // substringsStartPositions,
2114 // (substringsStartPositions = new int[substringsCount * 2]),
2115 // 0, substringsCount);
2116 // System.arraycopy(substringsEndPositions, 0,
2117 // (substringsEndPositions = new int[substringsCount * 2]),
2118 // 0, substringsCount);
2120 // if (splitOperatorsCount == splitOperators.length) {
2121 // System.arraycopy(splitOperators, 0,
2122 // (splitOperators = new int[splitOperatorsCount * 2]), 0,
2123 // splitOperatorsCount);
2125 // substringsStartPositions[substringsCount] = position;
2126 // substringsEndPositions[substringsCount++] = splitScanner.startPosition;
2127 // // substring ends on operator start
2128 // position = splitScanner.currentPosition;
2129 // // next substring will start from operator end
2130 // splitOperators[splitOperatorsCount++] = currentToken;
2134 // case TokenNameCOLON :
2136 // // see 1FK7C5R, we only split on a colon, when it is associated
2137 // // with a question-mark.
2138 // // indeed it might appear also behind a case statement, and we do
2139 // // not to break at this point.
2140 // if ((splitOperatorsCount == 0)
2141 // || splitOperators[splitOperatorsCount - 1] != TokenNameQUESTION) {
2144 // case TokenNameextends :
2145 // case TokenNameimplements :
2146 // //case TokenNamethrows :
2147 // case TokenNameDOT :
2149 // case TokenNameMULTIPLY :
2151 // case TokenNameDIVIDE :
2153 // case TokenNameREMAINDER :
2155 // case TokenNamePLUS :
2156 // // + (15.17, 15.17.2)
2157 // case TokenNameMINUS :
2159 // case TokenNameLEFT_SHIFT :
2161 // case TokenNameRIGHT_SHIFT :
2163 // // case TokenNameUNSIGNED_RIGHT_SHIFT : // >>> (15.18)
2164 // case TokenNameLESS :
2166 // case TokenNameLESS_EQUAL :
2168 // case TokenNameGREATER :
2170 // case TokenNameGREATER_EQUAL :
2172 // // case TokenNameinstanceof : // instanceof
2173 // case TokenNameEQUAL_EQUAL :
2174 // // == (15.20, 15.20.1, 15.20.2, 15.20.3)
2175 // case TokenNameEQUAL_EQUAL_EQUAL :
2176 // // == (15.20, 15.20.1, 15.20.2, 15.20.3)
2177 // case TokenNameNOT_EQUAL :
2178 // // != (15.20, 15.20.1, 15.20.2, 15.20.3)
2179 // case TokenNameNOT_EQUAL_EQUAL :
2180 // // != (15.20, 15.20.1, 15.20.2, 15.20.3)
2181 // case TokenNameAND :
2182 // // & (15.21, 15.21.1, 15.21.2)
2183 // case TokenNameOR :
2184 // // | (15.21, 15.21.1, 15.21.2)
2185 // case TokenNameXOR :
2186 // // ^ (15.21, 15.21.1, 15.21.2)
2187 // case TokenNameAND_AND :
2189 // case TokenNameOR_OR :
2191 // case TokenNameQUESTION :
2193 // case TokenNameMULTIPLY_EQUAL :
2195 // case TokenNameDIVIDE_EQUAL :
2197 // case TokenNameREMAINDER_EQUAL :
2199 // case TokenNamePLUS_EQUAL :
2201 // case TokenNameMINUS_EQUAL :
2203 // case TokenNameLEFT_SHIFT_EQUAL :
2205 // case TokenNameRIGHT_SHIFT_EQUAL :
2207 // // case TokenNameUNSIGNED_RIGHT_SHIFT_EQUAL : // >>>= (15.25.2)
2208 // case TokenNameAND_EQUAL :
2210 // case TokenNameXOR_EQUAL :
2212 // case TokenNameOR_EQUAL :
2214 // if ((openParenthesisPositionCount < splitTokenDepth || (openParenthesisPositionCount == splitTokenDepth && splitTokenPriority
2215 // > getTokenPriority(currentToken)))
2216 // && !((currentToken == TokenNamePLUS || currentToken == TokenNameMINUS) && (previousToken == TokenNameLBRACE
2217 // || previousToken == TokenNameLBRACKET || splitScanner.startPosition == 0))) {
2218 // // the current token is better than the one we currently have
2219 // // (in level or in priority if same level)
2220 // // reset the substringsCount
2221 // splitTokenDepth = openParenthesisPositionCount;
2222 // splitTokenType = currentToken;
2223 // splitTokenPriority = getTokenPriority(currentToken);
2224 // substringsStartPositions[0] = 0;
2225 // // better token means the whole line until now is the first
2227 // if (separateFirstArgumentOn(firstTokenOnLine)
2228 // && openParenthesisPositionCount > 0) {
2229 // substringsCount = 2; // resets the count of substrings
2230 // substringsEndPositions[0] = openParenthesisPosition[splitTokenDepth - 1];
2231 // substringsStartPositions[1] = openParenthesisPosition[splitTokenDepth - 1];
2232 // substringsEndPositions[1] = splitScanner.startPosition;
2233 // splitOperatorsCount = 3; // resets the count of split operators
2234 // splitOperators[0] = 0;
2235 // splitOperators[1] = 0;
2236 // splitOperators[2] = currentToken;
2237 // position = splitScanner.currentPosition;
2238 // // next substring will start from operator end
2240 // substringsCount = 1; // resets the count of substrings
2241 // substringsEndPositions[0] = splitScanner.startPosition;
2242 // // substring ends on operator start
2243 // position = splitScanner.currentPosition;
2244 // // next substring will start from operator end
2245 // splitOperatorsCount = 2; // resets the count of split operators
2246 // splitOperators[0] = 0;
2247 // // nothing for first operand since operator will be inserted in
2248 // // front of the second operand
2249 // splitOperators[1] = currentToken;
2252 // if (openParenthesisPositionCount == splitTokenDepth
2253 // && splitTokenPriority == getTokenPriority(currentToken)) {
2254 // // if another token with the same priority is found,
2255 // // push the start position of the substring and
2256 // // push the token into the stack.
2257 // // create a new array object if the current one is full.
2258 // if (substringsCount == substringsStartPositions.length) {
2261 // substringsStartPositions,
2263 // (substringsStartPositions = new int[substringsCount * 2]),
2264 // 0, substringsCount);
2265 // System.arraycopy(substringsEndPositions, 0,
2266 // (substringsEndPositions = new int[substringsCount * 2]),
2267 // 0, substringsCount);
2269 // if (splitOperatorsCount == splitOperators.length) {
2270 // System.arraycopy(splitOperators, 0,
2271 // (splitOperators = new int[splitOperatorsCount * 2]), 0,
2272 // splitOperatorsCount);
2274 // substringsStartPositions[substringsCount] = position;
2275 // substringsEndPositions[substringsCount++] = splitScanner.startPosition;
2276 // // substring ends on operator start
2277 // position = splitScanner.currentPosition;
2278 // // next substring will start from operator end
2279 // splitOperators[splitOperatorsCount++] = currentToken;
2285 // if (isComment(currentToken)) {
2286 // lastCommentStartPosition = splitScanner.startPosition;
2288 // lastCommentStartPosition = -1;
2291 // } catch (InvalidInputException e) {
2294 // // if the string cannot be split, return null.
2295 // if (splitOperatorsCount == 0)
2297 // // ## SPECIAL CASES BEGIN
2298 // if (((splitOperatorsCount == 2 && splitOperators[1] == TokenNameDOT
2299 // && splitTokenDepth == 0 && lastOpenParenthesisPosition > -1)
2300 // || (splitOperatorsCount > 2 && splitOperators[1] == TokenNameDOT
2301 // && splitTokenDepth == 0 && lastOpenParenthesisPosition > -1 && lastOpenParenthesisPosition <= options.maxLineLength) ||
2302 // (separateFirstArgumentOn(firstTokenOnLine)
2303 // && splitTokenDepth > 0 && lastOpenParenthesisPosition > -1))
2304 // && (lastOpenParenthesisPosition < splitScanner.source.length && splitScanner.source[lastOpenParenthesisPosition] != ')')) {
2305 // // fix for 1FH4J2H: LFCOM:WINNT - Formatter - Empty parenthesis should
2306 // // not be broken on two lines
2307 // // only one split on a top level .
2308 // // or more than one split on . and substring before open parenthesis fits
2310 // // or split inside parenthesis and first token is not a for/while/if
2311 // SplitLine sl = split(
2312 // stringToSplit.substring(lastOpenParenthesisPosition),
2313 // lastOpenParenthesisPosition);
2314 // if (sl == null || sl.operators[0] != TokenNameCOMMA) {
2315 // // trim() is used to remove the extra blanks at the end of the
2316 // // substring. See PR 1FGYPI1
2317 // return new SplitLine(new int[]{0, 0}, new String[]{
2318 // stringToSplit.substring(0, lastOpenParenthesisPosition).trim(),
2319 // stringToSplit.substring(lastOpenParenthesisPosition)}, new int[]{
2320 // offsetInGlobalLine,
2321 // lastOpenParenthesisPosition + offsetInGlobalLine});
2323 // // right substring can be split and is split on comma
2324 // // copy substrings and operators
2325 // // except if the 1st string is empty.
2326 // int startIndex = (sl.substrings[0].length() == 0) ? 1 : 0;
2327 // int subStringsLength = sl.substrings.length + 1 - startIndex;
2328 // String[] result = new String[subStringsLength];
2329 // int[] startIndexes = new int[subStringsLength];
2330 // int operatorsLength = sl.operators.length + 1 - startIndex;
2331 // int[] operators = new int[operatorsLength];
2332 // result[0] = stringToSplit.substring(0, lastOpenParenthesisPosition);
2333 // operators[0] = 0;
2334 // System.arraycopy(sl.startSubstringsIndexes, startIndex, startIndexes,
2335 // 1, subStringsLength - 1);
2336 // for (int i = subStringsLength - 1; i >= 0; i--) {
2337 // startIndexes[i] += offsetInGlobalLine;
2339 // System.arraycopy(sl.substrings, startIndex, result, 1,
2340 // subStringsLength - 1);
2341 // System.arraycopy(sl.operators, startIndex, operators, 1,
2342 // operatorsLength - 1);
2343 // return new SplitLine(operators, result, startIndexes);
2346 // // if the last token is a comment and the substring before the comment fits
2348 // // split before the comment and return the result.
2349 // if (lastCommentStartPosition > -1
2350 // && lastCommentStartPosition < options.maxLineLength
2351 // && splitTokenPriority > 50) {
2352 // int end = lastCommentStartPosition;
2353 // int start = lastCommentStartPosition;
2354 // if (stringToSplit.charAt(end - 1) == ' ') {
2357 // if (start != end && stringToSplit.charAt(start) == ' ') {
2360 // return new SplitLine(new int[]{0, 0}, new String[]{
2361 // stringToSplit.substring(0, end), stringToSplit.substring(start)},
2362 // new int[]{0, start});
2364 // if (position != stringToSplit.length()) {
2365 // if (substringsCount == substringsStartPositions.length) {
2366 // System.arraycopy(substringsStartPositions, 0,
2367 // (substringsStartPositions = new int[substringsCount * 2]), 0,
2368 // substringsCount);
2369 // System.arraycopy(substringsEndPositions, 0,
2370 // (substringsEndPositions = new int[substringsCount * 2]), 0,
2371 // substringsCount);
2373 // // avoid empty extra substring, e.g. line terminated with a semi-colon
2374 // substringsStartPositions[substringsCount] = position;
2375 // substringsEndPositions[substringsCount++] = stringToSplit.length();
2377 // if (splitOperatorsCount == splitOperators.length) {
2378 // System.arraycopy(splitOperators, 0,
2379 // (splitOperators = new int[splitOperatorsCount * 2]), 0,
2380 // splitOperatorsCount);
2382 // splitOperators[splitOperatorsCount] = 0;
2383 // // the last element of the stack is the position of the end of
2385 // // +1 because the substring method excludes the last character
2386 // String[] result = new String[substringsCount];
2387 // for (int i = 0; i < substringsCount; i++) {
2388 // int start = substringsStartPositions[i];
2389 // int end = substringsEndPositions[i];
2390 // if (stringToSplit.charAt(start) == ' ') {
2392 // substringsStartPositions[i]++;
2394 // if (end != start && stringToSplit.charAt(end - 1) == ' ') {
2397 // result[i] = stringToSplit.substring(start, end);
2398 // substringsStartPositions[i] += offsetInGlobalLine;
2400 // if (splitOperatorsCount > substringsCount) {
2401 // System.arraycopy(substringsStartPositions, 0,
2402 // (substringsStartPositions = new int[splitOperatorsCount]), 0,
2403 // substringsCount);
2404 // System.arraycopy(substringsEndPositions, 0,
2405 // (substringsEndPositions = new int[splitOperatorsCount]), 0,
2406 // substringsCount);
2407 // for (int i = substringsCount; i < splitOperatorsCount; i++) {
2408 // substringsStartPositions[i] = position;
2409 // substringsEndPositions[i] = position;
2411 // System.arraycopy(splitOperators, 0,
2412 // (splitOperators = new int[splitOperatorsCount]), 0,
2413 // splitOperatorsCount);
2415 // System.arraycopy(substringsStartPositions, 0,
2416 // (substringsStartPositions = new int[substringsCount]), 0,
2417 // substringsCount);
2418 // System.arraycopy(substringsEndPositions, 0,
2419 // (substringsEndPositions = new int[substringsCount]), 0,
2420 // substringsCount);
2421 // System.arraycopy(splitOperators, 0,
2422 // (splitOperators = new int[substringsCount]), 0, substringsCount);
2424 // SplitLine splitLine = new SplitLine(splitOperators, result,
2425 // substringsStartPositions);
2426 // return splitLine;
2429 private void updateMappedPositions(int startPosition) {
2430 if (positionsToMap == null) {
2433 char[] source = scanner.source;
2434 int sourceLength = source.length;
2435 while (indexToMap < positionsToMap.length && positionsToMap[indexToMap] <= startPosition) {
2436 int posToMap = positionsToMap[indexToMap];
2437 if (posToMap < 0 || posToMap >= sourceLength) {
2438 // protection against out of bounds position
2439 if (posToMap == sourceLength) {
2440 mappedPositions[indexToMap] = formattedSource.length();
2442 indexToMap = positionsToMap.length; // no more mapping
2445 if (CharOperation.isWhitespace(source[posToMap])) {
2446 mappedPositions[indexToMap] = startPosition + globalDelta + lineDelta;
2448 if (posToMap == sourceLength - 1) {
2449 mappedPositions[indexToMap] = startPosition + globalDelta + lineDelta;
2451 mappedPositions[indexToMap] = posToMap + globalDelta + lineDelta;
2458 private void updateMappedPositionsWhileSplitting(int startPosition, int endPosition) {
2459 if (mappedPositions == null || mappedPositions.length == indexInMap)
2461 while (indexInMap < mappedPositions.length && startPosition <= mappedPositions[indexInMap]
2462 && mappedPositions[indexInMap] < endPosition && indexInMap < indexToMap) {
2463 mappedPositions[indexInMap] += splitDelta;
2468 private int getLength(String s, int tabDepth) {
2470 for (int i = 0; i < tabDepth; i++) {
2471 length += options.tabSize;
2473 for (int i = 0, max = s.length(); i < max; i++) {
2474 char currentChar = s.charAt(i);
2475 switch (currentChar) {
2477 length += options.tabSize;
2487 * Sets the initial indentation level
2489 * @param indentationLevel
2490 * new indentation level
2494 public void setInitialIndentationLevel(int newIndentationLevel) {
2495 this.initialIndentationLevel = currentLineIndentationLevel = indentationLevel = newIndentationLevel;