/* * 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.intellij.openapi.vcs.changes.committed; import com.intellij.openapi.application.ApplicationManager; import com.intellij.openapi.progress.ProgressManager; import com.intellij.openapi.project.Project; import com.intellij.openapi.util.Couple; import com.intellij.openapi.vcs.*; import com.intellij.util.messages.MessageBusConnection; import org.jetbrains.annotations.NotNull; import java.util.Collections; import java.util.HashMap; import java.util.Map; public class RepositoryLocationCache { private final Project myProject; private final Map<Couple<String>, RepositoryLocation> myMap; public RepositoryLocationCache(final Project project) { myProject = project; myMap = Collections.synchronizedMap(new HashMap<Couple<String>, RepositoryLocation>()); final MessageBusConnection connection = myProject.getMessageBus().connect(); final VcsListener listener = new VcsListener() { @Override public void directoryMappingChanged() { reset(); } }; connection.subscribe(ProjectLevelVcsManager.VCS_CONFIGURATION_CHANGED, listener); connection.subscribe(ProjectLevelVcsManager.VCS_CONFIGURATION_CHANGED_IN_PLUGIN, listener); } public RepositoryLocation getLocation(final AbstractVcs vcs, final FilePath filePath, final boolean silent) { final Couple<String> key = Couple.of(vcs.getName(), filePath.getPath()); RepositoryLocation location = myMap.get(key); if (location != null) { return location; } location = getUnderProgress(vcs, filePath, silent); myMap.put(key, location); return location; } private RepositoryLocation getUnderProgress(final AbstractVcs vcs, final FilePath filePath, final boolean silent) { final MyLoader loader = new MyLoader(vcs, filePath); if ((! silent) && ApplicationManager.getApplication().isDispatchThread()) { ProgressManager.getInstance().runProcessWithProgressSynchronously(loader, "Discovering location of " + filePath.getPresentableUrl(), true, myProject); } else { loader.run(); } return loader.getLocation(); } public void reset() { myMap.clear(); } private class MyLoader implements Runnable { private final AbstractVcs myVcs; private final FilePath myFilePath; private RepositoryLocation myLocation; private MyLoader(@NotNull final AbstractVcs vcs, @NotNull FilePath filePath) { myVcs = vcs; myFilePath = filePath; } public void run() { final CommittedChangesProvider committedChangesProvider = myVcs.getCommittedChangesProvider(); if (committedChangesProvider != null) { myLocation = committedChangesProvider.getLocationFor(myFilePath); } } public RepositoryLocation getLocation() { return myLocation; } } }