| /* |
| * Copyright 2000-2009 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 org.jetbrains.idea.svn.checkin; |
| |
| import com.intellij.openapi.application.ApplicationManager; |
| import com.intellij.openapi.diagnostic.Logger; |
| import com.intellij.openapi.diff.impl.patch.formove.FilePathComparator; |
| import com.intellij.openapi.progress.ProgressIndicator; |
| import com.intellij.openapi.progress.ProgressManager; |
| import com.intellij.openapi.project.Project; |
| import com.intellij.openapi.ui.MessageType; |
| import com.intellij.openapi.util.Condition; |
| import com.intellij.openapi.util.Pair; |
| import com.intellij.openapi.vcs.CheckinProjectPanel; |
| import com.intellij.openapi.vcs.FilePath; |
| import com.intellij.openapi.vcs.VcsException; |
| import com.intellij.openapi.vcs.changes.Change; |
| import com.intellij.openapi.vcs.changes.ChangeList; |
| import com.intellij.openapi.vcs.changes.ChangesUtil; |
| import com.intellij.openapi.vcs.changes.ContentRevision; |
| import com.intellij.openapi.vcs.checkin.CheckinEnvironment; |
| import com.intellij.openapi.vcs.ui.RefreshableOnComponent; |
| import com.intellij.openapi.vcs.ui.VcsBalloonProblemNotifier; |
| import com.intellij.openapi.vfs.VfsUtilCore; |
| import com.intellij.openapi.vfs.VirtualFile; |
| import com.intellij.util.FunctionUtil; |
| import com.intellij.util.NullableFunction; |
| import com.intellij.util.PairConsumer; |
| import com.intellij.util.containers.ContainerUtil; |
| import com.intellij.util.containers.Convertor; |
| import com.intellij.util.containers.MultiMap; |
| import org.jetbrains.annotations.NotNull; |
| import org.jetbrains.annotations.Nullable; |
| import org.jetbrains.idea.svn.*; |
| import org.jetbrains.idea.svn.api.Depth; |
| import org.jetbrains.idea.svn.api.ProgressEvent; |
| import org.jetbrains.idea.svn.api.ProgressTracker; |
| import org.jetbrains.idea.svn.commandLine.SvnBindException; |
| import org.jetbrains.idea.svn.status.Status; |
| import org.jetbrains.idea.svn.status.StatusType; |
| import org.tmatesoft.svn.core.SVNException; |
| import org.tmatesoft.svn.core.SVNURL; |
| |
| import javax.swing.*; |
| import java.awt.*; |
| import java.io.File; |
| import java.util.*; |
| import java.util.List; |
| |
| public class SvnCheckinEnvironment implements CheckinEnvironment { |
| |
| private static final Logger LOG = Logger.getInstance(SvnCheckinEnvironment.class); |
| @NotNull private final SvnVcs mySvnVcs; |
| |
| public SvnCheckinEnvironment(@NotNull SvnVcs svnVcs) { |
| mySvnVcs = svnVcs; |
| } |
| |
| public RefreshableOnComponent createAdditionalOptionsPanel(CheckinProjectPanel panel, |
| PairConsumer<Object, Object> additionalDataConsumer) { |
| return new KeepLocksComponent(); |
| } |
| |
| @Nullable |
| public String getDefaultMessageFor(FilePath[] filesToCheckin) { |
| return null; |
| } |
| |
| @Nullable |
| public String getHelpId() { |
| return null; |
| } |
| |
| private List<VcsException> commitInt(List<File> paths, final String comment, final Set<String> feedback) { |
| final List<VcsException> exception = new ArrayList<VcsException>(); |
| final List<File> committables = getCommitables(paths); |
| final ProgressIndicator progress = ProgressManager.getInstance().getProgressIndicator(); |
| |
| if (progress != null) { |
| doCommit(committables, comment, exception, feedback); |
| } |
| else if (ApplicationManager.getApplication().isDispatchThread()) { |
| ProgressManager.getInstance().runProcessWithProgressSynchronously(new Runnable() { |
| public void run() { |
| doCommit(committables, comment, exception, feedback); |
| } |
| }, SvnBundle.message("progress.title.commit"), false, mySvnVcs.getProject()); |
| } |
| else { |
| doCommit(committables, comment, exception, feedback); |
| } |
| |
| return exception; |
| } |
| |
| private void doCommit(List<File> committables, String comment, List<VcsException> exception, final Set<String> feedback) { |
| //noinspection unchecked |
| final MultiMap<Pair<SVNURL,WorkingCopyFormat>,File> map = SvnUtil.splitIntoRepositoriesMap(mySvnVcs, committables, Convertor.SELF); |
| for (Map.Entry<Pair<SVNURL, WorkingCopyFormat>, Collection<File>> entry : map.entrySet()) { |
| try { |
| doCommitOneRepo(entry.getValue(), comment, exception, feedback, entry.getKey().getSecond()); |
| } |
| catch (VcsException e) { |
| LOG.info(e); |
| exception.add(e); |
| } |
| } |
| } |
| |
| private void doCommitOneRepo(Collection<File> committables, |
| String comment, |
| List<VcsException> exception, |
| final Set<String> feedback, |
| @NotNull WorkingCopyFormat format) |
| throws VcsException { |
| if (committables.isEmpty()) { |
| return; |
| } |
| |
| CommitInfo[] results = mySvnVcs.getFactory(format).createCheckinClient().commit(committables, comment); |
| |
| final StringBuilder committedRevisions = new StringBuilder(); |
| for (CommitInfo result : results) { |
| if (result.getErrorMessage() != null) { |
| exception.add(new VcsException(result.getErrorMessage().getFullMessage())); |
| } |
| else if (result != CommitInfo.EMPTY && result.getRevision() > 0) { |
| if (committedRevisions.length() > 0) { |
| committedRevisions.append(", "); |
| } |
| committedRevisions.append(result.getRevision()); |
| } |
| } |
| if (committedRevisions.length() > 0) { |
| reportCommittedRevisions(feedback, committedRevisions.toString()); |
| } |
| } |
| |
| private void reportCommittedRevisions(Set<String> feedback, String committedRevisions) { |
| final Project project = mySvnVcs.getProject(); |
| final String message = SvnBundle.message("status.text.comitted.revision", committedRevisions); |
| if (feedback == null) { |
| ApplicationManager.getApplication().invokeLater(new Runnable() { |
| public void run() { |
| new VcsBalloonProblemNotifier(project, message, MessageType.INFO).run(); |
| } |
| }, new Condition<Object>() { |
| @Override |
| public boolean value(Object o) { |
| return (! project.isOpen()) || project.isDisposed(); |
| } |
| }); |
| } else { |
| feedback.add("Subversion: " + message); |
| } |
| } |
| |
| private static class Adder { |
| private final List<File> myResult = new ArrayList<File>(); |
| private final Set<String> myDuplicatesControlSet = new HashSet<String>(); |
| |
| public void add(final File file) { |
| final String path = file.getAbsolutePath(); |
| if (! myDuplicatesControlSet.contains(path)) { |
| myResult.add(file); |
| myDuplicatesControlSet.add(path); |
| } |
| } |
| |
| public List<File> getResult() { |
| return myResult; |
| } |
| } |
| |
| private List<File> getCommitables(List<File> paths) { |
| final Adder adder = new Adder(); |
| |
| for (File path : paths) { |
| File file = path.getAbsoluteFile(); |
| adder.add(file); |
| if (file.getParentFile() != null) { |
| addParents(file.getParentFile(), adder); |
| } |
| } |
| return adder.getResult(); |
| } |
| |
| private void addParents(File file, final Adder adder) { |
| Status status = getStatus(file); |
| |
| if (status != null && status.is(StatusType.STATUS_ADDED, StatusType.STATUS_REPLACED)) { |
| // file should be added |
| adder.add(file); |
| file = file.getParentFile(); |
| if (file != null) { |
| addParents(file, adder); |
| } |
| } |
| } |
| |
| @Nullable |
| private Status getStatus(@NotNull File file) { |
| Status result = null; |
| |
| try { |
| result = mySvnVcs.getFactory(file).createStatusClient().doStatus(file, false); |
| } |
| catch (SvnBindException e) { |
| LOG.info(e); |
| } |
| |
| return result; |
| } |
| |
| private static List<File> collectPaths(@NotNull List<Change> changes) { |
| // case sensitive.. |
| Set<String> paths = ContainerUtil.newHashSet(); |
| |
| for (Change change : changes) { |
| addPath(paths, change.getBeforeRevision()); |
| addPath(paths, change.getAfterRevision()); |
| } |
| |
| return SvnUtil.toFiles(paths); |
| } |
| |
| private static void addPath(@NotNull Collection<String> paths, @Nullable ContentRevision revision) { |
| if (revision != null) { |
| paths.add(revision.getFile().getIOFile().getAbsolutePath()); |
| } |
| } |
| |
| public String getCheckinOperationName() { |
| return SvnBundle.message("checkin.operation.name"); |
| } |
| |
| public List<VcsException> commit(List<Change> changes, |
| String preparedComment, |
| @NotNull NullableFunction<Object, Object> parametersHolder, |
| Set<String> feedback) { |
| return commitInt(collectPaths(changes), preparedComment, feedback); |
| } |
| |
| public List<VcsException> commit(List<Change> changes, String preparedComment) { |
| return commit(changes, preparedComment, FunctionUtil.nullConstant(), null); |
| } |
| |
| public List<VcsException> scheduleMissingFileForDeletion(List<FilePath> filePaths) { |
| List<VcsException> exceptions = new ArrayList<VcsException>(); |
| List<File> files = ChangesUtil.filePathsToFiles(filePaths); |
| |
| for (File file : files) { |
| try { |
| mySvnVcs.getFactory(file).createDeleteClient().delete(file, true, false, null); |
| } |
| catch (VcsException e) { |
| exceptions.add(e); |
| } |
| } |
| |
| return exceptions; |
| } |
| |
| public List<VcsException> scheduleUnversionedFilesForAddition(List<VirtualFile> files) { |
| return scheduleUnversionedFilesForAddition(mySvnVcs, files); |
| } |
| |
| public static List<VcsException> scheduleUnversionedFilesForAddition(@NotNull SvnVcs vcs, List<VirtualFile> files) { |
| return scheduleUnversionedFilesForAddition(vcs, files, false); |
| } |
| |
| public static List<VcsException> scheduleUnversionedFilesForAddition(@NotNull SvnVcs vcs, List<VirtualFile> files, final boolean recursive) { |
| Collections.sort(files, FilePathComparator.getInstance()); |
| |
| ProgressTracker eventHandler = new SvnProgressCanceller() { |
| @Override |
| public void consume(ProgressEvent event) throws SVNException { |
| // TODO: indicator is null here when invoking "Add" action |
| ProgressIndicator indicator = ProgressManager.getInstance().getProgressIndicator(); |
| File file = event.getFile(); |
| |
| if (indicator != null && file != null) { |
| indicator.setText(SvnBundle.message("progress.text2.adding", file.getName() + " (" + file.getParent() + ")")); |
| } |
| } |
| }; |
| |
| List<VcsException> exceptions = new ArrayList<VcsException>(); |
| Depth depth = Depth.allOrEmpty(recursive); |
| |
| for (VirtualFile file : files) { |
| try { |
| File convertedFile = VfsUtilCore.virtualToIoFile(file); |
| |
| vcs.getFactory(convertedFile).createAddClient().add(convertedFile, depth, true, false, true, eventHandler); |
| } |
| catch (VcsException e) { |
| exceptions.add(e); |
| } |
| } |
| |
| return exceptions; |
| } |
| |
| public boolean keepChangeListAfterCommit(ChangeList changeList) { |
| return false; |
| } |
| |
| @Override |
| public boolean isRefreshAfterCommitNeeded() { |
| return true; |
| } |
| |
| private class KeepLocksComponent implements RefreshableOnComponent { |
| |
| @NotNull private final JCheckBox myKeepLocksBox; |
| private boolean myIsKeepLocks; |
| @NotNull private final JPanel myPanel; |
| @NotNull private final JCheckBox myAutoUpdate; |
| |
| public KeepLocksComponent() { |
| |
| myPanel = new JPanel(new BorderLayout()); |
| myKeepLocksBox = new JCheckBox(SvnBundle.message("checkbox.chckin.keep.files.locked")); |
| myKeepLocksBox.setSelected(myIsKeepLocks); |
| myAutoUpdate = new JCheckBox("Auto-update after commit"); |
| |
| myPanel.add(myAutoUpdate, BorderLayout.NORTH); |
| myPanel.add(myKeepLocksBox, BorderLayout.CENTER); |
| } |
| |
| public JComponent getComponent() { |
| return myPanel; |
| } |
| |
| public boolean isKeepLocks() { |
| return myKeepLocksBox.isSelected(); |
| } |
| |
| public boolean isAutoUpdate() { |
| return myAutoUpdate.isSelected(); |
| } |
| |
| public void refresh() { |
| } |
| |
| public void saveState() { |
| final SvnConfiguration configuration = mySvnVcs.getSvnConfiguration(); |
| configuration.setKeepLocks(isKeepLocks()); |
| configuration.setAutoUpdateAfterCommit(isAutoUpdate()); |
| } |
| |
| public void restoreState() { |
| final SvnConfiguration configuration = mySvnVcs.getSvnConfiguration(); |
| myIsKeepLocks = configuration.isKeepLocks(); |
| myAutoUpdate.setSelected(configuration.isAutoUpdateAfterCommit()); |
| } |
| } |
| } |