/* * 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. */ /* * Created by IntelliJ IDEA. * User: cdr * Date: Jul 13, 2007 * Time: 2:09:28 PM */ package com.intellij.psi.util.proximity; import com.intellij.extapi.psi.MetadataPsiElementBase; import com.intellij.openapi.module.Module; import com.intellij.openapi.module.ModuleUtil; import com.intellij.openapi.util.Computable; import com.intellij.openapi.util.Key; import com.intellij.psi.PsiElement; import com.intellij.psi.Weigher; import com.intellij.psi.WeighingComparable; import com.intellij.psi.WeighingService; import com.intellij.psi.statistics.StatisticsManager; import com.intellij.psi.util.ProximityLocation; import com.intellij.util.ProcessingContext; import com.intellij.util.containers.ContainerUtil; import com.intellij.util.containers.FactoryMap; import org.jetbrains.annotations.Nullable; import java.util.Comparator; public class PsiProximityComparator implements Comparator<Object> { public static final Key<ProximityStatistician> STATISTICS_KEY = Key.create("proximity"); public static final Key<ProximityWeigher> WEIGHER_KEY = Key.create("proximity"); private static final Weigher<PsiElement, ProximityLocation>[] PROXIMITY_WEIGHERS = ContainerUtil.toArray(WeighingService.getWeighers(WEIGHER_KEY), new Weigher[0]); private final PsiElement myContext; private final FactoryMap<PsiElement, WeighingComparable<PsiElement, ProximityLocation>> myProximities = new FactoryMap<PsiElement, WeighingComparable<PsiElement, ProximityLocation>>() { @Override protected WeighingComparable<PsiElement, ProximityLocation> create(final PsiElement key) { return getProximity(key, myContext); } }; private static final Key<Module> MODULE_BY_LOCATION = Key.create("ModuleByLocation"); public PsiProximityComparator(@Nullable PsiElement context) { myContext = context; } @Override public int compare(final Object o1, final Object o2) { PsiElement element1 = o1 instanceof PsiElement ? (PsiElement)o1 : null; PsiElement element2 = o2 instanceof PsiElement ? (PsiElement)o2 : null; if (element1 == null) return element2 == null ? 0 : 1; if (element2 == null) return -1; final WeighingComparable<PsiElement, ProximityLocation> proximity1 = myProximities.get(element1); final WeighingComparable<PsiElement, ProximityLocation> proximity2 = myProximities.get(element2); if (proximity1 == null || proximity2 == null) { return 0; } if (!proximity1.equals(proximity2)) { return - proximity1.compareTo(proximity2); } if (myContext == null) return 0; Module contextModule = ModuleUtil.findModuleForPsiElement(myContext); if (contextModule == null) return 0; StatisticsManager statisticsManager = StatisticsManager.getInstance(); final ProximityLocation location = new ProximityLocation(myContext, contextModule); int count1 = statisticsManager.getUseCount(STATISTICS_KEY, element1, location); int count2 = statisticsManager.getUseCount(STATISTICS_KEY, element1, location); return count2 - count1; } @Nullable public static WeighingComparable<PsiElement, ProximityLocation> getProximity(final PsiElement element, final PsiElement context) { if (element == null) return null; if (element instanceof MetadataPsiElementBase) return null; final Module contextModule = context != null ? ModuleUtil.findModuleForPsiElement(context) : null; return WeighingService.weigh(WEIGHER_KEY, element, new ProximityLocation(context, contextModule)); } @Nullable public static WeighingComparable<PsiElement, ProximityLocation> getProximity(final PsiElement element, final PsiElement context, ProcessingContext processingContext) { return getProximity(new Computable.PredefinedValueComputable<PsiElement>(element), context, processingContext); } @Nullable public static WeighingComparable<PsiElement, ProximityLocation> getProximity(final Computable<PsiElement> elementComputable, final PsiElement context, ProcessingContext processingContext) { PsiElement element = elementComputable.compute(); if (element == null) return null; if (element instanceof MetadataPsiElementBase) return null; if (context == null) return null; Module contextModule = processingContext.get(MODULE_BY_LOCATION); if (contextModule == null) { contextModule = ModuleUtil.findModuleForPsiElement(context); processingContext.put(MODULE_BY_LOCATION, contextModule); } if (contextModule == null) return null; return new WeighingComparable<PsiElement,ProximityLocation>(elementComputable, new ProximityLocation(context, contextModule, processingContext), PROXIMITY_WEIGHERS); } }