com.jetbrains.python.formatter.PyPreFormatProcessor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of python-community Show documentation
Show all versions of python-community Show documentation
A packaging of the IntelliJ Community Edition python-community library.
This is release number 1 of trunk branch 142.
The newest version!
/*
* Copyright 2000-2014 JetBrains s.r.o.
*
* 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.
*/
package com.jetbrains.python.formatter;
import com.intellij.lang.ASTNode;
import com.intellij.openapi.editor.Document;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.util.Couple;
import com.intellij.openapi.util.TextRange;
import com.intellij.openapi.util.text.StringUtil;
import com.intellij.psi.*;
import com.intellij.psi.codeStyle.CodeStyleSettings;
import com.intellij.psi.codeStyle.CodeStyleSettingsManager;
import com.intellij.psi.impl.source.codeStyle.PreFormatProcessor;
import com.jetbrains.python.PythonLanguage;
import com.jetbrains.python.psi.LanguageLevel;
import com.jetbrains.python.psi.PyElementGenerator;
import com.jetbrains.python.psi.PyRecursiveElementVisitor;
import org.jetbrains.annotations.NotNull;
import java.util.ArrayList;
import java.util.List;
/**
* @author traff
*/
public class PyPreFormatProcessor implements PreFormatProcessor {
@NotNull
@Override
public TextRange process(@NotNull ASTNode element, @NotNull TextRange range) {
PsiElement psiElement = element.getPsi();
if (psiElement == null) return range;
if (!psiElement.getLanguage().is(PythonLanguage.getInstance())) return range;
PsiFile file = psiElement.isValid() ? psiElement.getContainingFile() : null;
if (file == null) return range;
Project project = psiElement.getProject();
return new PyCommentFormatter(project).process(psiElement, range);
}
/**
* @author traff
*/
public static class PyCommentFormatter extends PyRecursiveElementVisitor {
private final Project myProject;
private final CodeStyleSettings mySettings;
private final PyCodeStyleSettings myPyCodeStyleSettings;
private final List> myCommentReplacements = new ArrayList>();
private TextRange myRange;
private int myDelta = 0;
public PyCommentFormatter(Project project) {
myProject = project;
mySettings = CodeStyleSettingsManager.getSettings(project);
myPyCodeStyleSettings = mySettings.getCustomSettings(PyCodeStyleSettings.class);
}
public TextRange process(PsiElement element, TextRange range) {
if (!myPyCodeStyleSettings.SPACE_AFTER_NUMBER_SIGN) {
return range;
}
myRange = range;
final PsiDocumentManager manager = PsiDocumentManager.getInstance(myProject);
final Document document = manager.getDocument(element.getContainingFile());
if (document != null) {
manager.doPostponedOperationsAndUnblockDocument(document);
try {
// collect all comments
element.accept(this);
for (Couple pair : myCommentReplacements) {
pair.getFirst().replace(pair.getSecond());
}
}
finally {
manager.commitDocument(document);
}
}
return TextRange.create(range.getStartOffset(), range.getEndOffset() + myDelta);
}
@Override
public void visitComment(PsiComment comment) {
if (!myRange.contains(comment.getTextRange())) {
return;
}
final String origText = comment.getText();
final int commentStart = origText.indexOf('#');
if (commentStart != -1 && (commentStart + 1) < origText.length()) {
final char charAfterDash = origText.charAt(commentStart + 1);
if (charAfterDash == '!' && comment.getTextRange().getStartOffset() == 0) {
return; // shebang
}
if (charAfterDash == '#' || charAfterDash == ':') {
return; // doc comment
}
final String commentTextWithoutDash = origText.substring(commentStart + 1);
final String newText;
if (isTrailingComment(comment)) {
newText = "# " + StringUtil.trimLeading(commentTextWithoutDash);
}
else if (!StringUtil.isWhiteSpace(charAfterDash)) {
newText = "# " + commentTextWithoutDash;
}
else {
return;
}
if (!newText.equals(origText)) {
myDelta += newText.length() - origText.length();
final PyElementGenerator elementGenerator = PyElementGenerator.getInstance(myProject);
final PsiComment newComment = elementGenerator.createFromText(LanguageLevel.forElement(comment), PsiComment.class, newText);
myCommentReplacements.add(Couple.of(comment, newComment));
}
}
}
}
private static boolean isTrailingComment(@NotNull PsiComment comment) {
final PsiElement prevElement = comment.getPrevSibling();
return prevElement != null && (!(prevElement instanceof PsiWhiteSpace) || !prevElement.textContains('\n'));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy