blob: 8ec833e8258bc843a322caced273ba6057694dbe [file] [log] [blame]
/*
* Copyright 2000-2012 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.intellij.codeInsight.actions;
import com.intellij.application.options.editor.EditorOptions;
import com.intellij.codeInsight.CodeInsightBundle;
import com.intellij.find.impl.FindInProjectUtil;
import com.intellij.formatting.FormattingModelBuilder;
import com.intellij.ide.util.PropertiesComponent;
import com.intellij.lang.LanguageFormatting;
import com.intellij.notification.Notification;
import com.intellij.notification.NotificationListener;
import com.intellij.notification.NotificationType;
import com.intellij.notification.Notifications;
import com.intellij.openapi.actionSystem.*;
import com.intellij.openapi.application.ApplicationManager;
import com.intellij.openapi.command.CommandProcessor;
import com.intellij.openapi.components.ServiceManager;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.editor.Editor;
import com.intellij.openapi.editor.SelectionModel;
import com.intellij.openapi.editor.ex.EditorSettingsExternalizable;
import com.intellij.openapi.module.Module;
import com.intellij.openapi.options.ShowSettingsUtil;
import com.intellij.openapi.project.DumbAware;
import com.intellij.openapi.project.DumbService;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.util.TextRange;
import com.intellij.openapi.vfs.ReadonlyStatusHandler;
import com.intellij.openapi.vfs.VirtualFile;
import com.intellij.openapi.wm.IdeFrame;
import com.intellij.openapi.wm.ex.WindowManagerEx;
import com.intellij.psi.*;
import com.intellij.psi.codeStyle.arrangement.engine.ArrangementEngine;
import com.intellij.psi.search.GlobalSearchScope;
import com.intellij.psi.search.LocalSearchScope;
import com.intellij.psi.search.SearchScope;
import com.intellij.psi.util.PsiUtilCore;
import org.jetbrains.annotations.NonNls;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.annotations.TestOnly;
import javax.swing.*;
import javax.swing.event.HyperlinkEvent;
import java.util.ArrayList;
import java.util.Collections;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
public class ReformatCodeAction extends AnAction implements DumbAware {
private static final Logger LOG = Logger.getInstance("#com.intellij.codeInsight.actions.ReformatCodeAction");
private static final @NonNls String HELP_ID = "editing.codeReformatting";
protected static ReformatFilesOptions myTestOptions;
@Override
public void actionPerformed(AnActionEvent event) {
DataContext dataContext = event.getDataContext();
final Project project = CommonDataKeys.PROJECT.getData(dataContext);
if (project == null) {
return;
}
PsiDocumentManager.getInstance(project).commitAllDocuments();
final Editor editor = CommonDataKeys.EDITOR.getData(dataContext);
final VirtualFile[] files = CommonDataKeys.VIRTUAL_FILE_ARRAY.getData(dataContext);
PsiFile file = null;
final PsiDirectory dir;
boolean hasSelection = false;
if (editor != null){
file = PsiDocumentManager.getInstance(project).getPsiFile(editor.getDocument());
if (file == null) return;
dir = file.getContainingDirectory();
hasSelection = editor.getSelectionModel().hasSelection();
}
else if (areFiles(files)) {
final ReadonlyStatusHandler.OperationStatus operationStatus = ReadonlyStatusHandler.getInstance(project).ensureFilesWritable(files);
if (!operationStatus.hasReadonlyFiles()) {
ReformatFilesOptions selectedFlags = getReformatFilesOptions(project, files);
if (selectedFlags == null)
return;
final boolean processOnlyChangedText = selectedFlags.isProcessOnlyChangedText();
final boolean shouldOptimizeImports = selectedFlags.isOptimizeImports() && !DumbService.getInstance(project).isDumb();
AbstractLayoutCodeProcessor processor = new ReformatCodeProcessor(project, convertToPsiFiles(files, project), null, processOnlyChangedText);
if (shouldOptimizeImports) {
processor = new OptimizeImportsProcessor(processor);
}
if (selectedFlags.isRearrangeEntries()) {
processor = new RearrangeCodeProcessor(processor, null);
}
processor.run();
}
return;
}
else {
Project projectContext = PlatformDataKeys.PROJECT_CONTEXT.getData(dataContext);
Module moduleContext = LangDataKeys.MODULE_CONTEXT.getData(dataContext);
if (projectContext != null || moduleContext != null) {
ReformatFilesOptions selectedFlags = getLayoutProjectOptions(project, moduleContext); // module menu - only 2 options available
if (selectedFlags != null) {
reformatModule(project, moduleContext, selectedFlags);
}
return;
}
PsiElement element = CommonDataKeys.PSI_ELEMENT.getData(dataContext);
if (element == null) return;
if (element instanceof PsiDirectoryContainer) {
dir = ((PsiDirectoryContainer)element).getDirectories()[0];
}
else if (element instanceof PsiDirectory) {
dir = (PsiDirectory)element;
}
else {
file = element.getContainingFile();
if (file == null) return;
dir = file.getContainingDirectory();
}
}
boolean optimizeImports = ReformatFilesDialog.isOptmizeImportsOptionOn();
boolean processWholeFile = false;
boolean processChangedTextOnly = PropertiesComponent.getInstance().getBoolean(LayoutCodeConstants.PROCESS_CHANGED_TEXT_KEY, false);
boolean rearrangeEntries = getLastSavedRearrangeCbState(project, file);
final boolean showDialog = EditorSettingsExternalizable.getInstance().getOptions().SHOW_REFORMAT_DIALOG;
if (file == null && dir != null) {
DirectoryFormattingOptions options = getDirectoryFormattingOptions(project, dir);
if (options != null) {
reformatDirectory(project, dir, options);
}
return;
}
else if (showDialog) {
LayoutCodeOptions selectedFlags = getLayoutCodeOptions(project, file, dir, hasSelection);
if (selectedFlags == null)
return;
optimizeImports = selectedFlags.isOptimizeImports();
rearrangeEntries = selectedFlags.isRearrangeEntries();
processWholeFile = selectedFlags.isProcessWholeFile();
processChangedTextOnly = selectedFlags.isProcessOnlyChangedText();
if (selectedFlags.isProcessDirectory()) {
assert dir != null : "File = " + file + ", Element = " + CommonDataKeys.PSI_ELEMENT.getData(dataContext);
reformatDirectory(project, dir, selectedFlags);
return;
}
}
final TextRange range;
final boolean processSelectedText = !processWholeFile && hasSelection;
if (processSelectedText) {
range = TextRange.create(editor.getSelectionModel().getSelectionStart(), editor.getSelectionModel().getSelectionEnd());
}
else{
range = null;
}
if (optimizeImports && range == null) {
new OptimizeImportsProcessor(new ReformatCodeProcessor(project, file, null, processChangedTextOnly)).run();
}
else {
new ReformatCodeProcessor(project, file, range, !processSelectedText && processChangedTextOnly).run();
}
if (rearrangeEntries && file != null && editor != null) {
final ArrangementEngine engine = ServiceManager.getService(project, ArrangementEngine.class);
try {
final PsiFile finalFile = file;
SelectionModel selectionModel = editor.getSelectionModel();
final TextRange rangeToUse = selectionModel.hasSelection()
? TextRange.create(selectionModel.getSelectionStart(), selectionModel.getSelectionEnd())
: TextRange.create(0, editor.getDocument().getTextLength());
CommandProcessor.getInstance().executeCommand(project, new Runnable() {
@Override
public void run() {
engine.arrange(editor, finalFile, Collections.singleton(rangeToUse));
}
}, getTemplatePresentation().getText(), null);
}
finally {
PsiDocumentManager.getInstance(project).commitDocument(editor.getDocument());
}
}
}
@Nullable
private static DirectoryFormattingOptions getDirectoryFormattingOptions(@NotNull Project project, @NotNull PsiDirectory dir) {
LayoutDirectoryDialog dialog = new LayoutDirectoryDialog(
project,
CodeInsightBundle.message("process.reformat.code"),
CodeInsightBundle.message("process.scope.directory", dir.getVirtualFile().getPath()),
FormatChangedTextUtil.hasChanges(dir)
);
boolean enableIncludeDirectoriesCb = dir.getSubdirectories().length > 0;
dialog.setEnabledIncludeSubdirsCb(enableIncludeDirectoriesCb);
dialog.setSelectedIncludeSubdirsCb(enableIncludeDirectoriesCb);
dialog.show();
if (dialog.isOK()) {
return dialog;
}
return null;
}
private static void reformatDirectory(@NotNull Project project,
@NotNull PsiDirectory dir,
@NotNull DirectoryFormattingOptions options)
{
AbstractLayoutCodeProcessor processor = new ReformatCodeProcessor(
project,
dir,
options.isIncludeSubdirectories(),
options.isProcessOnlyChangedText()
);
registerScopeFilter(processor, options.getSearchScope());
registerFileMaskFilter(processor, options.getFileTypeMask());
if (options.isOptimizeImports()) {
processor = new OptimizeImportsProcessor(processor);
}
if (options.isRearrangeEntries()) {
processor = new RearrangeCodeProcessor(processor, null);
}
processor.run();
}
private static void reformatModule(@NotNull Project project,
@Nullable Module moduleContext,
@NotNull ReformatFilesOptions selectedFlags)
{
boolean shouldOptimizeImports = selectedFlags.isOptimizeImports() && !DumbService.getInstance(project).isDumb();
boolean processOnlyChangedText = selectedFlags.isProcessOnlyChangedText();
AbstractLayoutCodeProcessor processor;
if (moduleContext != null)
processor = new ReformatCodeProcessor(project, moduleContext, processOnlyChangedText);
else
processor = new ReformatCodeProcessor(project, processOnlyChangedText);
registerScopeFilter(processor, selectedFlags.getSearchScope());
registerFileMaskFilter(processor, selectedFlags.getFileTypeMask());
if (shouldOptimizeImports) {
processor = new OptimizeImportsProcessor(processor);
}
if (selectedFlags.isRearrangeEntries()) {
processor = new RearrangeCodeProcessor(processor, null);
}
processor.run();
}
public static void registerScopeFilter(@NotNull AbstractLayoutCodeProcessor processor, @Nullable final SearchScope scope) {
if (scope == null) {
return;
}
processor.addFileFilter(new FileFilter() {
@Override
public boolean accept(@NotNull VirtualFile file) {
if (scope instanceof LocalSearchScope) {
return ((LocalSearchScope)scope).isInScope(file);
}
if (scope instanceof GlobalSearchScope) {
return ((GlobalSearchScope)scope).contains(file);
}
return false;
}
});
}
public static void registerFileMaskFilter(@NotNull AbstractLayoutCodeProcessor processor, @Nullable String fileTypeMask) {
if (fileTypeMask == null)
return;
final Pattern pattern = getFileTypeMaskPattern(fileTypeMask);
if (pattern != null) {
processor.addFileFilter(new FileFilter() {
@Override
public boolean accept(@NotNull VirtualFile file) {
return pattern.matcher(file.getName()).matches();
}
});
}
}
@Nullable
private static Pattern getFileTypeMaskPattern(@Nullable String mask) {
try {
return FindInProjectUtil.createFileMaskRegExp(mask);
} catch (PatternSyntaxException e) {
LOG.info("Error while processing file mask: ", e);
return null;
}
}
public static void updateShowDialogSetting(LayoutCodeDialog dialog, String title) {
if (dialog.isDoNotAskMe()) {
Notifications.Bus.notify(new Notification("Reformat Code", title,
"<html>You can re-enable the dialog on the <a href=''>IDE Settings -> Editor</a> pane</html>",
NotificationType.INFORMATION, new NotificationListener() {
@Override
public void hyperlinkUpdate(@NotNull Notification notification, @NotNull HyperlinkEvent e) {
if (e.getEventType() == HyperlinkEvent.EventType.ACTIVATED) {
final ShowSettingsUtil util = ShowSettingsUtil.getInstance();
IdeFrame ideFrame = WindowManagerEx.getInstanceEx().findFrameFor(null);
util.editConfigurable((JFrame)ideFrame, new EditorOptions());
}
}
}));
}
}
public static PsiFile[] convertToPsiFiles(final VirtualFile[] files,Project project) {
final PsiManager manager = PsiManager.getInstance(project);
final ArrayList<PsiFile> result = new ArrayList<PsiFile>();
for (VirtualFile virtualFile : files) {
final PsiFile psiFile = manager.findFile(virtualFile);
if (psiFile != null) result.add(psiFile);
}
return PsiUtilCore.toPsiFileArray(result);
}
@Override
public void update(AnActionEvent event){
Presentation presentation = event.getPresentation();
DataContext dataContext = event.getDataContext();
Project project = CommonDataKeys.PROJECT.getData(dataContext);
if (project == null){
presentation.setEnabled(false);
return;
}
Editor editor = CommonDataKeys.EDITOR.getData(dataContext);
final VirtualFile[] files = CommonDataKeys.VIRTUAL_FILE_ARRAY.getData(dataContext);
if (editor != null){
PsiFile file = PsiDocumentManager.getInstance(project).getPsiFile(editor.getDocument());
if (file == null || file.getVirtualFile() == null) {
presentation.setEnabled(false);
return;
}
if (LanguageFormatting.INSTANCE.forContext(file) != null) {
presentation.setEnabled(true);
return;
}
}
else if (files!= null && areFiles(files)) {
boolean anyFormatters = false;
for (VirtualFile virtualFile : files) {
if (virtualFile.isDirectory()) {
presentation.setEnabled(false);
return;
}
final PsiFile psiFile = PsiManager.getInstance(project).findFile(virtualFile);
if (psiFile == null) {
presentation.setEnabled(false);
return;
}
final FormattingModelBuilder builder = LanguageFormatting.INSTANCE.forContext(psiFile);
if (builder != null) {
anyFormatters = true;
}
}
if (!anyFormatters) {
presentation.setEnabled(false);
return;
}
}
else if (files != null && files.length == 1) {
// skip. Both directories and single files are supported.
}
else if (LangDataKeys.MODULE_CONTEXT.getData(dataContext) == null &&
PlatformDataKeys.PROJECT_CONTEXT.getData(dataContext) == null) {
PsiElement element = CommonDataKeys.PSI_ELEMENT.getData(dataContext);
if (element == null) {
presentation.setEnabled(false);
return;
}
if (!(element instanceof PsiDirectory)) {
PsiFile file = element.getContainingFile();
if (file == null || LanguageFormatting.INSTANCE.forContext(file) == null) {
presentation.setEnabled(false);
return;
}
}
}
presentation.setEnabled(true);
}
@Nullable
private static ReformatFilesOptions getReformatFilesOptions(@NotNull Project project, @NotNull VirtualFile[] files) {
if (ApplicationManager.getApplication().isUnitTestMode()) {
return myTestOptions;
}
ReformatFilesDialog dialog = new ReformatFilesDialog(project, files);
dialog.show();
if (!dialog.isOK()) return null;
return dialog;
}
@Nullable
private static ReformatFilesOptions getLayoutProjectOptions(@NotNull Project project, @Nullable Module module) {
if (ApplicationManager.getApplication().isUnitTestMode()) {
return myTestOptions;
}
final String text = module != null ? CodeInsightBundle.message("process.scope.module", module.getModuleFilePath())
: CodeInsightBundle.message("process.scope.project", project.getPresentableUrl());
final boolean enableOnlyVCSChangedRegions = module != null ? FormatChangedTextUtil.hasChanges(module)
: FormatChangedTextUtil.hasChanges(project);
LayoutProjectCodeDialog dialog = new LayoutProjectCodeDialog(project, CodeInsightBundle.message("process.reformat.code"), text, enableOnlyVCSChangedRegions);
dialog.show();
if (!dialog.isOK()) return null;
return dialog;
}
@Nullable
private static LayoutCodeOptions getLayoutCodeOptions(@NotNull Project project,
@Nullable PsiFile file,
@Nullable PsiDirectory dir,
boolean hasSelection)
{
if (ApplicationManager.getApplication().isUnitTestMode()) {
return (LayoutCodeOptions)myTestOptions;
}
LayoutCodeDialog dialog = new LayoutCodeDialog(project, CodeInsightBundle.message("process.reformat.code"),
file, dir, hasSelection ? Boolean.TRUE : Boolean.FALSE, HELP_ID);
dialog.show();
if (!dialog.isOK()) return null;
EditorSettingsExternalizable.getInstance().getOptions().SHOW_REFORMAT_DIALOG = !dialog.isDoNotAskMe();
updateShowDialogSetting(dialog, "\"Reformat Code\" dialog disabled");
return dialog;
}
public static boolean getLastSavedRearrangeCbState(@NotNull Project project, @Nullable PsiFile file) {
if (file != null) {
return LayoutCodeSettingsStorage.getLastSavedRearrangeEntriesCbStateFor(project, file.getLanguage());
}
return LayoutCodeSettingsStorage.getLastSavedRearrangeEntriesCbStateFor(project);
}
@TestOnly
protected static void setTestOptions(ReformatFilesOptions options) {
myTestOptions = options;
}
public static boolean areFiles(final VirtualFile[] files) {
if (files == null) return false;
if (files.length < 2) return false;
for (VirtualFile virtualFile : files) {
if (virtualFile.isDirectory()) return false;
}
return true;
}
}