mirror of
https://github.com/swc-project/swc.git
synced 2024-12-27 15:42:51 +03:00
741 lines
35 KiB
TypeScript
741 lines
35 KiB
TypeScript
//
|
||
// Copyright (c) Microsoft Corporation. All rights reserved.
|
||
//
|
||
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
// you may not use this file except in compliance with the License.
|
||
// You may obtain a copy of the License at
|
||
// http://www.apache.org/licenses/LICENSE-2.0
|
||
//
|
||
// Unless required by applicable law or agreed to in writing, software
|
||
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
// See the License for the specific language governing permissions and
|
||
// limitations under the License.
|
||
//
|
||
|
||
///<reference path='formatting.ts' />
|
||
|
||
|
||
module Formatting {
|
||
export class Indenter implements ILineIndenationResolver {
|
||
|
||
private indentationBag: IndentationBag;
|
||
private scriptBlockBeginLineNumber: number;
|
||
private offsetIndentationDeltas: Dictionary_int_int;
|
||
|
||
constructor(
|
||
public logger: TypeScript.ILogger,
|
||
public tree: ParseTree,
|
||
public snapshot: ITextSnapshot,
|
||
public languageHostIndentation: string,
|
||
public editorOptions: Services.EditorOptions,
|
||
public firstToken: TokenSpan,
|
||
public smartIndent: boolean) {
|
||
|
||
this.indentationBag = new IndentationBag(this.snapshot);
|
||
this.scriptBlockBeginLineNumber = -1;
|
||
this.offsetIndentationDeltas = new Dictionary_int_int(); // text offset -> indentation delta
|
||
|
||
// by default the root (program) has zero indendation
|
||
this.tree.Root.SetIndentationOverride("");
|
||
|
||
this.ApplyScriptBlockIndentation(this.languageHostIndentation, this.tree);
|
||
this.FillInheritedIndentation(this.tree);
|
||
|
||
}
|
||
|
||
public GetIndentationEdits(token: TokenSpan, nextToken: TokenSpan, node: ParseNode, sameLineIndent: boolean): List_TextEditInfo {
|
||
if (this.logger.information()) {
|
||
this.logger.log("GetIndentationEdits(" +
|
||
"t1=[" + token.Span.startPosition() + "," + token.Span.endPosition()+ "], " +
|
||
"t2=[" + (nextToken == null ? "null" : (nextToken.Span.startPosition() + "," + nextToken.Span.endPosition())) + "]" +
|
||
")");
|
||
}
|
||
|
||
var result = this.GetIndentationEditsWorker(token, nextToken, node, sameLineIndent);
|
||
|
||
if (this.logger.information()) {
|
||
for (var i = 0; i < result.count() ; i++) {
|
||
var edit = result.get(i);
|
||
this.logger.log("edit: minChar=" + edit.position + ", limChar=" + (edit.position + edit.length) + ", text=\"" + TypeScript.stringToLiteral(edit.replaceWith, 30) + "\"");
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
public GetIndentationEditsWorker(token: TokenSpan, nextToken: TokenSpan, node: ParseNode, sameLineIndent: boolean): List_TextEditInfo {
|
||
var result = new List_TextEditInfo();
|
||
var indentationInfo: IndentationInfo = null;
|
||
|
||
// This handles the case:
|
||
// return (
|
||
// function() {
|
||
// })
|
||
// The given function's node indicates that the function starts directly after "return (".
|
||
// In this case, we adjust the span to point to the function keyword.
|
||
// The same applies to objects and arrays.
|
||
// The reason this is done inside the Indenter is because it only affects indentation behavior.
|
||
// It's also done in ParseTree when we traverse up the tree because we don't have the
|
||
// tokens for nodes outside the span we are formatting.
|
||
this.AdjustStartOffsetIfNeeded(token, node);
|
||
|
||
// Don't adjust indentation on the same line of a script block
|
||
if (this.scriptBlockBeginLineNumber == token.lineNumber()) {
|
||
return result;
|
||
}
|
||
|
||
// Don't indent multi-line strings
|
||
if (!sameLineIndent && this.IsMultiLineString(token)) {
|
||
return result;
|
||
}
|
||
|
||
// Special cases for the tokens that don't show up in the tree, such as curly braces and comments
|
||
indentationInfo = this.GetSpecialCaseIndentation(token, node);
|
||
if (indentationInfo == null) {
|
||
//// For anything else
|
||
|
||
// Get the indentation level only from the node that starts on the same offset as the token
|
||
// otherwise the token is not meant to be indented
|
||
while (!node.CanIndent() && node.Parent != null && token.Span.span.start() == node.Parent.AuthorNode.Details.StartOffset)
|
||
node = node.Parent;
|
||
|
||
if (node.CanIndent() && token.Span.span.start() == node.AuthorNode.Details.StartOffset) {
|
||
indentationInfo = node.GetEffectiveIndentation(this);
|
||
}
|
||
else {
|
||
//// Special cases for anything else that is not in the tree and should be indented
|
||
|
||
// check for label (identifier followed by a colon)
|
||
if (token.Token == AuthorTokenKind.atkIdentifier && nextToken != null && nextToken.Token == AuthorTokenKind.atkColon) {
|
||
// This will make the label on the same level as the surrounding function/block
|
||
// ex:
|
||
// {
|
||
// statement;
|
||
// label:
|
||
// statement;
|
||
// }
|
||
indentationInfo = node.GetEffectiveChildrenIndentation(this);
|
||
}
|
||
else {
|
||
//// Move the token the same indentation-delta that moved its indentable parent
|
||
//// For example:
|
||
//// var a,
|
||
//// b;
|
||
//// The declaration 'b' would remain under 'a' even if 'var' got indented.
|
||
indentationInfo = this.ApplyIndentationDeltaFromParent(token, node);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Get the indent edit from the indentation info
|
||
if (indentationInfo != null) {
|
||
var edit = this.GetIndentEdit(indentationInfo, token.Span.startPosition(), sameLineIndent);
|
||
if (edit != null) {
|
||
this.RegisterIndentation(edit, sameLineIndent);
|
||
|
||
result.add(edit);
|
||
|
||
// multi-line comments, apply delta indentation to all the other lines
|
||
if (token.Token == AuthorTokenKind.atkComment) {
|
||
var commentEdits = this.GetCommentIndentationEdits(token);
|
||
commentEdits.foreach((item) => {
|
||
result.add(item);
|
||
});
|
||
}
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
private GetCommentIndentationEdits(token: TokenSpan): List_TextEditInfo {
|
||
var result = new List_TextEditInfo();
|
||
|
||
if (token.Token != AuthorTokenKind.atkComment)
|
||
return result;
|
||
|
||
var commentLastLineNumber = this.snapshot.GetLineNumberFromPosition(token.Span.endPosition());
|
||
if (token.lineNumber() == commentLastLineNumber)
|
||
return result;
|
||
|
||
var commentFirstLineIndentationDelta = this.GetIndentationDelta(token.Span.startPosition(), null);
|
||
if (commentFirstLineIndentationDelta != undefined) {
|
||
for (var line = token.lineNumber() + 1; line <= commentLastLineNumber; line++) {
|
||
var lineStartPosition = this.snapshot.GetLineFromLineNumber(line).startPosition();
|
||
var lineIndent = this.GetLineIndentationForOffset(lineStartPosition);
|
||
|
||
var commentIndentationInfo = this.ApplyIndentationDelta2(lineIndent, commentFirstLineIndentationDelta);
|
||
if (commentIndentationInfo != null) {
|
||
var tokenStartPosition = lineStartPosition + lineIndent.length;
|
||
var commentIndentationEdit = this.GetIndentEdit(commentIndentationInfo, tokenStartPosition, false);
|
||
if (commentIndentationEdit != null) {
|
||
result.add(commentIndentationEdit);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
static GetIndentSizeFromIndentText(indentText: string, editorOptions: Services.EditorOptions): number {
|
||
return GetIndentSizeFromText(indentText, editorOptions, /*includeNonIndentChars:*/ false);
|
||
}
|
||
|
||
static GetIndentSizeFromText(text: string, editorOptions: Services.EditorOptions, includeNonIndentChars: boolean): number {
|
||
var indentSize = 0;
|
||
|
||
for (var i = 0; i < text.length; i++) {
|
||
var c = text.charAt(i);
|
||
|
||
if (c == '\t')
|
||
indentSize = (indentSize + editorOptions.TabSize) - (indentSize % editorOptions.TabSize);
|
||
else if (c == ' ')
|
||
indentSize += 1;
|
||
else {
|
||
if (includeNonIndentChars)
|
||
indentSize += 1;
|
||
else
|
||
break;
|
||
}
|
||
}
|
||
|
||
return indentSize;
|
||
}
|
||
|
||
private GetSpecialCaseIndentation(token: TokenSpan, node: ParseNode): IndentationInfo {
|
||
var indentationInfo: IndentationInfo = null;
|
||
|
||
switch (token.Token) {
|
||
case AuthorTokenKind.atkLCurly: // { is not part of the tree
|
||
indentationInfo = this.GetSpecialCaseIndentationForLCurly(node);
|
||
return indentationInfo;
|
||
|
||
case AuthorTokenKind.atkElse: // else is not part of the tree
|
||
case AuthorTokenKind.atkRBrack: // ] is not part of the tree
|
||
indentationInfo = node.GetNodeStartLineIndentation(this);
|
||
return indentationInfo;
|
||
|
||
case AuthorTokenKind.atkRCurly: // } is not part of the tree
|
||
// if '}' is for a body-block, get indentation based on its parent.
|
||
if (node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkBlock && node.AuthorNode.EdgeLabel == AuthorParseNodeEdge.apneBody)
|
||
node = node.Parent;
|
||
indentationInfo = node.GetNodeStartLineIndentation(this);
|
||
return indentationInfo;
|
||
|
||
case AuthorTokenKind.atkWhile: // while (in do-while) is not part of the tree
|
||
if (node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkDoWhile) {
|
||
indentationInfo = node.GetNodeStartLineIndentation(this);
|
||
return indentationInfo;
|
||
}
|
||
|
||
return null;
|
||
|
||
case AuthorTokenKind.atkSColon:
|
||
return this.GetSpecialCaseIndentationForSemicolon(token, node);
|
||
|
||
case AuthorTokenKind.atkComment:
|
||
return this.GetSpecialCaseIndentationForComment(token, node);
|
||
|
||
default:
|
||
return indentationInfo;
|
||
}
|
||
}
|
||
|
||
private GetSpecialCaseIndentationForLCurly(node: ParseNode): IndentationInfo {
|
||
var indentationInfo: IndentationInfo = null;
|
||
|
||
if (node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkFncDecl ||
|
||
node.AuthorNode.EdgeLabel == AuthorParseNodeEdge.apneThen || node.AuthorNode.EdgeLabel == AuthorParseNodeEdge.apneElse) {
|
||
// flushed with the node (function & if)
|
||
indentationInfo = node.GetNodeStartLineIndentation(this);
|
||
return indentationInfo;
|
||
}
|
||
else if (node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkObject && !node.CanIndent()) {
|
||
// if the open curly belongs to a non-indented object, do nothing here.
|
||
return null;
|
||
}
|
||
|
||
// effective identation of the block
|
||
indentationInfo = node.GetEffectiveIndentation(this);
|
||
return indentationInfo;
|
||
}
|
||
|
||
private GetSpecialCaseIndentationForSemicolon(token: TokenSpan, node: ParseNode): IndentationInfo {
|
||
var indentationInfo: IndentationInfo = null;
|
||
|
||
if (this.smartIndent) {
|
||
indentationInfo = node.GetEffectiveChildrenIndentation(this);
|
||
return indentationInfo;
|
||
}
|
||
else {
|
||
// Indent all semicolons except the ones that belong to the for statement parts (initalizer, condition, itnrement)
|
||
if (node.AuthorNode.Details.Kind != AuthorParseNodeKind.apnkFor) {
|
||
// The passed node is actually either the program or the list because semicolon doesn't belong
|
||
// to any statement in the tree, though the statement extends up to the semicolon position.
|
||
// To find the correct statement, we look for the adjacent node on the left of the semicolon.
|
||
var semiColonStartSpan = new Span(token.Span.startPosition(), 0);
|
||
node = ParseTree.FindCommonParentNode(semiColonStartSpan, semiColonStartSpan, node);
|
||
indentationInfo = node.GetEffectiveChildrenIndentation(this);
|
||
return indentationInfo;
|
||
}
|
||
}
|
||
|
||
return null;
|
||
}
|
||
|
||
private GetSpecialCaseIndentationForComment(token: TokenSpan, node: ParseNode): IndentationInfo {
|
||
var indentationInfo: IndentationInfo = null;
|
||
|
||
// Only indent line comment and the first line of block comment
|
||
var twoCharSpan = token.Span.Intersection(new Span(token.Span.startPosition(), 2));
|
||
if (twoCharSpan != null && (twoCharSpan.GetText() == "//" || twoCharSpan.GetText() == "/*")) {
|
||
while (node.ChildrenIndentationDelta == null && node.Parent != null)
|
||
node = node.Parent;
|
||
|
||
if (this.CanIndentComment(token, node)) {
|
||
indentationInfo = node.GetEffectiveChildrenIndentationForComment(this);
|
||
}
|
||
else {
|
||
indentationInfo = this.ApplyIndentationDeltaFromParent(token, node);
|
||
}
|
||
}
|
||
|
||
return indentationInfo;
|
||
}
|
||
|
||
private CanIndentComment(token: TokenSpan, node: ParseNode): boolean {
|
||
switch (node.AuthorNode.Details.Kind) {
|
||
case AuthorParseNodeKind.apnkProg:
|
||
case AuthorParseNodeKind.apnkBlock:
|
||
case AuthorParseNodeKind.apnkSwitch:
|
||
case AuthorParseNodeKind.apnkCase:
|
||
case AuthorParseNodeKind.apnkDefaultCase:
|
||
case AuthorParseNodeKind.apnkIf:
|
||
case AuthorParseNodeKind.apnkFor:
|
||
case AuthorParseNodeKind.apnkForIn:
|
||
case AuthorParseNodeKind.apnkWhile:
|
||
case AuthorParseNodeKind.apnkWith:
|
||
case AuthorParseNodeKind.apnkDoWhile:
|
||
case AuthorParseNodeKind.apnkObject:
|
||
return true;
|
||
|
||
case AuthorParseNodeKind.apnkFncDecl:
|
||
// Comments before arguments are not indented.
|
||
// This code doesn't cover the cases of comment after the last argument or
|
||
// when there are no arguments. Though this is okay since the only case we care about is:
|
||
// function foo(/* test */ a,
|
||
// /* test */ b)
|
||
var result = true;
|
||
var children = ParseNodeExtensions.FindChildrenWithEdge(node, AuthorParseNodeEdge.apneArgument);
|
||
children.foreach((argumentNode) => {
|
||
if (token.Span.startPosition() < argumentNode.AuthorNode.Details.StartOffset)
|
||
result = false;
|
||
});
|
||
|
||
return result;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
private ApplyScriptBlockIndentation(languageHostIndentation: string, tree: ParseTree): void
|
||
{
|
||
if (languageHostIndentation == null || tree.StartNodeSelf == null)
|
||
return;
|
||
|
||
var scriptBlockIndentation = this.ApplyIndentationLevel(languageHostIndentation, 1);
|
||
|
||
//TypeScript: Projection snapshots not supported
|
||
|
||
// Disconnect the sibling node if it belongs to a different script block
|
||
//IProjectionSnapshot projectionSnapshot = this.snapshot as IProjectionSnapshot;
|
||
//if (projectionSnapshot != null)
|
||
//{
|
||
// // Get script block spans.
|
||
// foreach (SnapshotSpan sourceSpan in projectionSnapshot.GetSourceSpans())
|
||
// {
|
||
// // Map the spans to the JavaScript buffer.
|
||
// ReadOnlyCollection<Span> spans = projectionSnapshot.MapFromSourceSnapshot(sourceSpan);
|
||
|
||
// Debug.Assert(spans.Count == 1, string.Format(CultureInfo.InvariantCulture, "Unexpected span count of {0}.", spans.Count));
|
||
|
||
// if (spans.Count > 0)
|
||
// {
|
||
// Span span = spans.First();
|
||
|
||
// // If the "self" node is the first root-level node in a script block, then remove the start node.
|
||
// if (span.Contains(tree.StartNodethis.AuthorNode.Details.StartOffset))
|
||
// {
|
||
// this.scriptBlockBeginLineNumber = projectionSnapshot.GetLineNumberFromPosition(span.Start);
|
||
|
||
// if (tree.StartNodePreviousSibling.HasValue)
|
||
// {
|
||
// int siblingStartOffset = tree.StartNodePreviousSibling.Value.Details.StartOffset;
|
||
|
||
// // Don't consider sibling in these cases:
|
||
// // 1. The sibling belongs to another script block
|
||
// // 2. The sibling is on the same line of the script block
|
||
// if (!span.Contains(siblingStartOffset) || projectionSnapshot.GetLineNumberFromPosition(siblingStartOffset) == this.scriptBlockBeginLineNumber)
|
||
// {
|
||
// tree.StartNodePreviousSibling = null;
|
||
// }
|
||
// }
|
||
|
||
// break;
|
||
// }
|
||
// }
|
||
// }
|
||
//}
|
||
|
||
// The root is the program.
|
||
tree.Root.SetIndentationOverride(scriptBlockIndentation);
|
||
}
|
||
|
||
private GetIndentEdit(indentInfo: IndentationInfo, tokenStartPosition: number, sameLineIndent: boolean): TextEditInfo {
|
||
var indentText = this.ApplyIndentationLevel(indentInfo.Prefix, indentInfo.Level);
|
||
|
||
if (sameLineIndent) {
|
||
return new TextEditInfo(tokenStartPosition, 0, indentText);
|
||
}
|
||
else {
|
||
var snapshotLine = this.snapshot.GetLineFromPosition(tokenStartPosition);
|
||
var currentIndentSpan = new Span(snapshotLine.startPosition(), tokenStartPosition - snapshotLine.startPosition());
|
||
var currentIndentText = this.snapshot.GetText(currentIndentSpan);
|
||
|
||
if (currentIndentText !== indentText) {
|
||
if (this.logger.debug()) {
|
||
// Verify that currentIndentText is all whitespaces
|
||
for (var i = 0, len = currentIndentText.length; i < len; i++) {
|
||
var c = currentIndentText.charCodeAt(i);
|
||
if (!StringUtils.IsWhiteSpace(c)) {
|
||
Debug.Fail("Formatting error: Will remove user code when indenting the line: " + snapshotLine.getText());
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
return new TextEditInfo(currentIndentSpan.start(), currentIndentSpan.length(), indentText);
|
||
}
|
||
}
|
||
|
||
return null;
|
||
}
|
||
|
||
private ApplyIndentationLevel(existingIndentation: string, level: number): string {
|
||
var indentSize = this.editorOptions.IndentSize;
|
||
var tabSize = this.editorOptions.TabSize;
|
||
var convertTabsToSpaces = this.editorOptions.ConvertTabsToSpaces;
|
||
|
||
if (level < 0) {
|
||
if (StringUtils.IsNullOrEmpty(existingIndentation))
|
||
return "";
|
||
|
||
var totalIndent = 0;
|
||
StringUtils.foreach(existingIndentation, (c) => {
|
||
if (c == '\t')
|
||
totalIndent += tabSize;
|
||
else
|
||
totalIndent++;
|
||
});
|
||
|
||
totalIndent += level * indentSize;
|
||
if (totalIndent < 0)
|
||
return "";
|
||
else
|
||
return this.GetIndentString(null, totalIndent, tabSize, convertTabsToSpaces);
|
||
}
|
||
|
||
var totalIndentSize = level * indentSize;
|
||
return this.GetIndentString(existingIndentation, totalIndentSize, tabSize, convertTabsToSpaces);
|
||
}
|
||
|
||
private GetIndentString(prefix: string, totalIndentSize: number, tabSize: number, convertTabsToSpaces: boolean): string {
|
||
var tabString = convertTabsToSpaces ? StringUtils.create(' ', tabSize) : "\t";
|
||
|
||
var text = "";
|
||
if (!StringUtils.IsNullOrEmpty(prefix))
|
||
text += prefix;
|
||
|
||
var pos = 0;
|
||
|
||
// fill first with tabs
|
||
while (pos <= totalIndentSize - tabSize) {
|
||
text += tabString;
|
||
pos += tabSize;
|
||
}
|
||
|
||
// fill the reminder with spaces
|
||
while (pos < totalIndentSize) {
|
||
text += ' ';
|
||
pos++;
|
||
}
|
||
|
||
return text;
|
||
}
|
||
|
||
private ApplyIndentationDeltaFromParent(token: TokenSpan, node: ParseNode): IndentationInfo {
|
||
var indentationInfo: IndentationInfo = null;
|
||
|
||
var indentableParent = node;
|
||
while (indentableParent != null && !indentableParent.CanIndent())
|
||
indentableParent = indentableParent.Parent;
|
||
|
||
if (indentableParent != null && indentableParent.AuthorNode.Details.Kind != AuthorParseNodeKind.apnkProg) {
|
||
var parentIndentationDeltaSize = this.GetIndentationDelta(indentableParent.AuthorNode.Details.StartOffset, token.Span.startPosition());
|
||
if (parentIndentationDeltaSize !== undefined) {
|
||
indentationInfo = this.ApplyIndentationDelta1(token.Span.startPosition(), parentIndentationDeltaSize);
|
||
}
|
||
}
|
||
|
||
return indentationInfo;
|
||
}
|
||
|
||
private ApplyIndentationDelta1(tokenStartPosition: number, delta: number): IndentationInfo {
|
||
// Get current indentation
|
||
var snapshotLine = this.snapshot.GetLineFromPosition(tokenStartPosition);
|
||
var currentIndentSpan = new Span(snapshotLine.startPosition(), tokenStartPosition - snapshotLine.startPosition());
|
||
var currentIndent = this.snapshot.GetText(currentIndentSpan);
|
||
|
||
// Calculate new indentation from current-indentation and delta
|
||
return this.ApplyIndentationDelta2(currentIndent, delta);
|
||
}
|
||
|
||
private ApplyIndentationDelta2(currentIndent: string, delta: number): IndentationInfo {
|
||
if (delta == 0)
|
||
return null;
|
||
|
||
var currentIndentSize = Indenter.GetIndentSizeFromIndentText(currentIndent, this.editorOptions);
|
||
|
||
var newIndentSize = currentIndentSize + delta;
|
||
if (newIndentSize < 0) {
|
||
newIndentSize = 0;
|
||
}
|
||
|
||
var newIndent = this.GetIndentString(null, newIndentSize, this.editorOptions.TabSize, this.editorOptions.ConvertTabsToSpaces);
|
||
if (newIndent != null) {
|
||
return new IndentationInfo(newIndent, 0);
|
||
}
|
||
|
||
return null;
|
||
}
|
||
|
||
private GetIndentationDelta(tokenStartPosition: number, childTokenStartPosition: number/*?*/): number/*?*/ {
|
||
Debug.Assert(childTokenStartPosition !== undefined, "Error: caller must pass 'null' for undefined position");
|
||
|
||
var indentationDeltaSize = this.offsetIndentationDeltas.GetValue(tokenStartPosition);
|
||
if (indentationDeltaSize === null) {
|
||
var indentEditInfo = this.indentationBag.FindIndent(tokenStartPosition);
|
||
|
||
// No recorded indentation, return null
|
||
if (indentEditInfo == null)
|
||
return null;
|
||
|
||
var origIndentText = this.snapshot.GetText(new Span(indentEditInfo.OrigIndentPosition, indentEditInfo.OrigIndentLength()));
|
||
var newIndentText = indentEditInfo.Indentation();
|
||
|
||
var origIndentSize = Indenter.GetIndentSizeFromText(origIndentText, this.editorOptions, /*includeNonIndentChars*/true);
|
||
var newIndentSize = Indenter.GetIndentSizeFromIndentText(newIndentText, this.editorOptions);
|
||
|
||
// Check the child's position whether it's before the parent position
|
||
// if so indent the child based on the first token on the line as opposed to the parent position
|
||
//
|
||
// Example of relative to parent (not line), relative indentation should be "4 (newIndentSize) - 9 (indentSize up to for) = -5"
|
||
//
|
||
// if (1) { for (i = 0; i < 10; => if (1) {
|
||
// i++) { for (i = 0; i < 10;
|
||
// i++) {
|
||
//
|
||
// Example of relative to line, relative indentation should be "4 (newIndentSize) - 0 (indentSize up to if) = 4"
|
||
//
|
||
// if (1) { for (i = 0; i < 10; => if (1) {
|
||
// i++) { for (i = 0; i < 10;
|
||
// i++) {
|
||
if (childTokenStartPosition !== null) {
|
||
var childTokenLineStartPosition = this.snapshot.GetLineFromPosition(childTokenStartPosition).startPosition();
|
||
var childIndentText = this.snapshot.GetText(new Span(childTokenLineStartPosition, childTokenStartPosition - childTokenLineStartPosition));
|
||
|
||
var childIndentSize = Indenter.GetIndentSizeFromIndentText(childIndentText, this.editorOptions);
|
||
|
||
if (childIndentSize < origIndentSize)
|
||
origIndentSize = Indenter.GetIndentSizeFromIndentText(origIndentText, this.editorOptions);
|
||
}
|
||
|
||
indentationDeltaSize = newIndentSize - origIndentSize;
|
||
this.offsetIndentationDeltas.Add(tokenStartPosition, indentationDeltaSize);
|
||
}
|
||
|
||
return indentationDeltaSize;
|
||
}
|
||
|
||
private FillInheritedIndentation(tree: ParseTree): void
|
||
{
|
||
var offset = -1;
|
||
var indentNode: ParseNode = null;
|
||
|
||
if (tree.StartNodeSelf != null) {
|
||
if (!this.smartIndent && tree.StartNodePreviousSibling !== null && tree.StartNodeSelf.AuthorNode.Label == 0 && tree.StartNodePreviousSibling.Label == 0) {
|
||
indentNode = tree.StartNodeSelf;
|
||
offset = tree.StartNodePreviousSibling.Details.StartOffset;
|
||
|
||
// In case the sibling node is on the same line of a parent node, ex:
|
||
// case 1: a++;
|
||
// break;
|
||
// In this example, the sibling of break is a++ but a++ is on the same line of its parent.
|
||
var lineNum = this.snapshot.GetLineNumberFromPosition(offset);
|
||
var node = indentNode;
|
||
while (node.Parent != null && this.snapshot.GetLineNumberFromPosition(node.Parent.AuthorNode.Details.StartOffset) == lineNum) {
|
||
node = node.Parent;
|
||
if (node.CanIndent()) {
|
||
indentNode = node;
|
||
indentNode.IndentationDelta = 0;
|
||
}
|
||
}
|
||
}
|
||
else {
|
||
var parent: ParseNode;
|
||
|
||
// Otherwise base on parent indentation.
|
||
if (this.smartIndent) {
|
||
// in smartIndent the self node is the parent node since it's the closest node to the new line
|
||
// ... unless in case if the startNodeSelf represents the firstToken then we need to choose its parent
|
||
parent = tree.StartNodeSelf;
|
||
while (parent != null && parent.AuthorNode.Details.StartOffset == this.firstToken.Span.startPosition())
|
||
parent = parent.Parent;
|
||
}
|
||
else {
|
||
// Get the parent that is really on a different line from the self node
|
||
var startNodeLineNumber = this.snapshot.GetLineNumberFromPosition(tree.StartNodeSelf.AuthorNode.Details.StartOffset);
|
||
parent = tree.StartNodeSelf.Parent;
|
||
while (parent != null &&
|
||
startNodeLineNumber == this.snapshot.GetLineNumberFromPosition(parent.AuthorNode.Details.StartOffset)) {
|
||
parent = parent.Parent;
|
||
}
|
||
}
|
||
|
||
// The parent node to take its indentation is the first parent that has indentation.
|
||
while (parent != null && !parent.CanIndent()) {
|
||
parent = parent.Parent;
|
||
}
|
||
|
||
// Skip Program since it has no indentation
|
||
if (parent != null && parent.AuthorNode.Details.Kind != AuthorParseNodeKind.apnkProg) {
|
||
offset = parent.AuthorNode.Details.StartOffset;
|
||
indentNode = parent;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (indentNode != null) {
|
||
var indentOverride = this.GetLineIndentationForOffset(offset);
|
||
|
||
// Set the indentation on all the siblings to be the same as indentNode
|
||
if (!this.smartIndent && tree.StartNodePreviousSibling !== null && indentNode.Parent != null) {
|
||
ParseNodeExtensions.GetChildren(indentNode.Parent).foreach((sibling) => {
|
||
if (sibling !== indentNode) {
|
||
if (sibling.CanIndent())
|
||
sibling.SetIndentationOverride(indentOverride);
|
||
}
|
||
});
|
||
}
|
||
|
||
// Set the indent override string on the indent node and on every parent (on different line) after adjusting the indent by the negative delta
|
||
var lastDelta = 0;
|
||
var lastLine = this.snapshot.GetLineNumberFromPosition(indentNode.AuthorNode.Details.StartOffset);
|
||
do {
|
||
var currentLine = this.snapshot.GetLineNumberFromPosition(indentNode.AuthorNode.Details.StartOffset);
|
||
if (lastLine != currentLine) {
|
||
lastLine = currentLine;
|
||
indentOverride = this.ApplyIndentationLevel(indentOverride, -lastDelta);
|
||
lastDelta = 0;
|
||
}
|
||
|
||
if (indentNode.CanIndent()) {
|
||
indentNode.SetIndentationOverride(indentOverride);
|
||
lastDelta = indentNode.IndentationDelta;
|
||
}
|
||
|
||
indentNode = indentNode.Parent;
|
||
}
|
||
while (indentNode != null);
|
||
}
|
||
}
|
||
|
||
public GetLineIndentationForOffset(offset: number): string {
|
||
var indentationEdit: IndentationEditInfo;
|
||
|
||
// First check if we already have indentation info in our indentation bag
|
||
indentationEdit = this.indentationBag.FindIndent(offset);
|
||
if (indentationEdit != null) {
|
||
return indentationEdit.Indentation();
|
||
}
|
||
else {
|
||
// Otherwise, use the indentation from the textBuffer
|
||
var line = this.snapshot.GetLineFromPosition(offset);
|
||
var lineText = line.getText();
|
||
var index = 0;
|
||
|
||
while (index < lineText.length && (lineText.charAt(index) == ' ' || lineText.charAt(index) == '\t')) {
|
||
++index;
|
||
}
|
||
|
||
return lineText.substr(0, index);
|
||
}
|
||
}
|
||
|
||
private RegisterIndentation(indent: TextEditInfo, sameLineIndent: boolean): void
|
||
{
|
||
var indentationInfo: IndentationEditInfo = null;
|
||
|
||
if (sameLineIndent) {
|
||
// Consider the original indentation from the beginning of the line up to the indent position (or really the token position)
|
||
var lineStartPosition = this.snapshot.GetLineFromPosition(indent.Position).startPosition();
|
||
var lineIndentLength = indent.Position - lineStartPosition;
|
||
|
||
indentationInfo = IndentationEditInfo.create2(indent.Position, indent.ReplaceWith, lineStartPosition, lineIndentLength);
|
||
}
|
||
else {
|
||
indentationInfo = new IndentationEditInfo(indent);
|
||
}
|
||
|
||
this.indentationBag.AddIndent(indentationInfo);
|
||
}
|
||
|
||
public RegisterIndentation2(position: number, indent: string): void
|
||
{
|
||
this.RegisterIndentation(new TextEditInfo(position, 0, indent), false);
|
||
}
|
||
|
||
private AdjustStartOffsetIfNeeded(token: TokenSpan, node: ParseNode): void
|
||
{
|
||
if (token == null)
|
||
return;
|
||
|
||
var updateStartOffset = false;
|
||
|
||
switch (token.Token) {
|
||
case AuthorTokenKind.atkFunction:
|
||
updateStartOffset = node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkFncDecl;
|
||
break;
|
||
|
||
case AuthorTokenKind.atkLCurly:
|
||
updateStartOffset = node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkObject;
|
||
break;
|
||
|
||
case AuthorTokenKind.atkLBrack:
|
||
updateStartOffset = node.AuthorNode.Details.Kind == AuthorParseNodeKind.apnkArray;
|
||
break;
|
||
}
|
||
|
||
if (updateStartOffset) {
|
||
ParseNodeExtensions.SetNodeSpan(node, token.Span.startPosition(), node.AuthorNode.Details.EndOffset);
|
||
}
|
||
}
|
||
|
||
private IsMultiLineString(token: TokenSpan): boolean {
|
||
return token.tokenID === TypeScript.TokenID.StringLiteral &&
|
||
this.snapshot.GetLineNumberFromPosition(token.Span.endPosition()) > this.snapshot.GetLineNumberFromPosition(token.Span.startPosition());
|
||
}
|
||
}
|
||
}
|