/* * 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. */ /* * @author max */ package com.intellij.psi; import com.intellij.openapi.util.Key; import com.intellij.openapi.util.KeyWithDefaultValue; import gnu.trove.THashMap; import java.util.Map; @SuppressWarnings({"unchecked", "ConstantConditions"}) public class ResolveState { private static final ResolveState ourInitialState = new ResolveState(); public static ResolveState initial() { return ourInitialState; } public <T> ResolveState put(Key<T> key, T value) { return new OneElementResolveState(key, value); } public <T> T get(Key<T> key) { if (key instanceof KeyWithDefaultValue) { return ((KeyWithDefaultValue<T>)key).getDefaultValue(); } return null; } private static class OneElementResolveState extends ResolveState { final Key myKey; final Object myValue; OneElementResolveState(Key key, Object value) { myKey = key; myValue = value; } @Override public <T> ResolveState put(Key<T> key, T value) { if (myKey.equals(key)) { return new OneElementResolveState(key, value); } return new TwoElementResolveState(myKey, myValue, key, value); } @Override public <T> T get(Key<T> key) { Object value = myKey.equals(key) ? myValue : null; if (value == null && key instanceof KeyWithDefaultValue) { return ((KeyWithDefaultValue<T>)key).getDefaultValue(); } return (T)value; } } private static class TwoElementResolveState extends ResolveState { final Key myKey1; final Object myValue1; final Key myKey2; final Object myValue2; TwoElementResolveState(Key key1, Object value1, Key key2, Object value2) { myKey1 = key1; myValue1 = value1; myKey2 = key2; myValue2 = value2; } @Override public <T> ResolveState put(Key<T> key, T value) { if (myKey1.equals(key)) { return new TwoElementResolveState(key, value, myKey2, myValue2); } if (myKey2.equals(key)) { return new TwoElementResolveState(myKey1, myValue1, key, value); } return new ManyElementResolveState(this, key, value); } @Override public <T> T get(Key<T> key) { Object value; if (myKey1.equals(key)) { value = myValue1; } else if (myKey2.equals(key)) { value = myValue2; } else { value = null; } if (value == null && key instanceof KeyWithDefaultValue) { return ((KeyWithDefaultValue<T>)key).getDefaultValue(); } return (T)value; } } private static class ManyElementResolveState extends ResolveState { private final Map<Object, Object> myValues = new THashMap<Object, Object>(); ManyElementResolveState(ManyElementResolveState parent, Key key, Object value) { myValues.putAll(parent.myValues); myValues.put(key, value); } ManyElementResolveState(TwoElementResolveState twoState, Key key, Object value) { myValues.put(twoState.myKey1, twoState.myValue1); myValues.put(twoState.myKey2, twoState.myValue2); myValues.put(key, value); } @Override public <T> ResolveState put(Key<T> key, T value) { return new ManyElementResolveState(this, key, value); } @Override public <T> T get(Key<T> key) { final T value = (T)myValues.get(key); if (value == null && key instanceof KeyWithDefaultValue) { return ((KeyWithDefaultValue<T>) key).getDefaultValue(); } return value; } } }