All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.openide.nodes.CookieSet Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.openide.nodes;

import java.lang.ref.Reference;
import java.lang.ref.WeakReference;

import java.util.*;

import javax.swing.event.ChangeListener;
import org.openide.nodes.Node.Cookie;
import org.openide.util.ChangeSupport;
import org.openide.util.Lookup;
import org.openide.util.lookup.AbstractLookup;
import org.openide.util.lookup.InstanceContent;


/** Support class for storing cookies and
* retriving them by representation class.
* Provides simple notifications about changes
* in cookies.
*
* @author Jaroslav Tulach
*/
public final class CookieSet extends Object implements Lookup.Provider {
    /** variable to allow effecient communication with NodeLookup, Node.Cookie or Class or Set */
    private static ThreadLocal QUERY_MODE = new ThreadLocal();

    /** list of cookies (Class, Node.Cookie) */
    private HashMap map = new HashMap(31);

    private final ChangeSupport cs = new ChangeSupport(this);
    
    /** potential instance content */
    private final CookieSetLkp ic;
    /** lookup to use return from the cookie set, if initialized */
    private Lookup lookup;

    /** Default constructor. */
    public CookieSet() {
        this(null, null);
    }
    
    private CookieSet(CookieSetLkp ic, Lookup lookup) {
        this.ic = ic;
        this.lookup = lookup;
    }
    
    /** Factory method to create new, general purpose cookie set. 
     * The general purpose means that it is possible to store
     * any object, into the cookie set and then obtain it using {@link #getLookup}
     * and queries on the returned {@link Lookup}. The before object can
     * be passed in if one wants to do a lazy initialization of the {@link CookieSet}
     * content.
     * 
     * @param before the interface to support lazy initialization
     * @return new cookie set that can contain not only {@link Node.Cookie} but also 
     *    any plain old java object
     * @see #assign
     * @since 7.0
     */
    public static CookieSet createGeneric(Before before) {
        CookieSetLkp al = new CookieSetLkp(before);
        return new CookieSet(al, al);
    }

    /** The lookup associated with this cookie set. Keeps track of
     * the same things that are in the cookie set, but presents them
     * as being inside the lookup.
     * 
     * @return the lookup representing this cookie set
     * @since 7.0
     */
    public Lookup getLookup() {
        synchronized (QUERY_MODE) {
            if (lookup == null) {
                AbstractNode an = new AbstractNode(this);
                lookup = an.getLookup();
            }
        }
        return lookup;
    }

    
    /** Add a new cookie to the set. If a cookie of the same
    * actual (not representation!) class is already there,
    * it is replaced.
    * 

Cookies inserted earlier are given preference during lookup, * in case a supplied representation class matches more than one cookie * in the set. * * @param cookie cookie to add */ public void add(Node.Cookie cookie) { addImpl((Object)cookie); fireChangeEvent(); } private void addImpl(Object cookie) { synchronized (this) { registerCookie(cookie.getClass(), cookie); } if (ic != null) { ic.add(cookie); } } /** Remove a cookie from the set. * @param cookie the cookie to remove */ public void remove(Node.Cookie cookie) { removeImpl((Object)cookie); fireChangeEvent(); } void removeImpl(Object cookie) { synchronized (this) { unregisterCookie(cookie.getClass(), cookie); } if (ic != null) { ic.remove(cookie); } } /** Get a cookie. * * @param clazz the representation class * @return a cookie assignable to the representation class, or null if there is none */ public T getCookie(Class clazz) { if (ic != null) { ic.beforeLookupImpl(clazz); } return lookupCookie(clazz); } private T lookupCookie(Class clazz) { Node.Cookie ret = null; Object queryMode = QUERY_MODE.get(); synchronized (this) { R r = findR(clazz); if (r == null) { if (queryMode == null || ic == null) { return null; } } else { ret = r.cookie(); if (queryMode instanceof Set) { @SuppressWarnings("unchecked") Set keys = (Set)queryMode; keys.addAll(map.keySet()); } } } if (ret instanceof CookieEntry) { if (clazz == queryMode) { // we expected to be asked for this class // set cookie entry as a result QUERY_MODE.set(ret); ret = null; } else { // unwrap the cookie ret = ((CookieEntry) ret).getCookie(true); } } else if (ret == null) { if (ic != null && (!Node.Cookie.class.isAssignableFrom(clazz) || clazz == Node.Cookie.class) ) { enhancedQueryMode(lookup, clazz); ret = null; } } return clazz.cast(ret); } static void enhancedQueryMode(Lookup lookup, Class clazz) { Object type = QUERY_MODE.get(); if (type != clazz) { return; } Collection> items = lookup.lookupResult(clazz).allItems(); if (items.size() == 0) { return; } AbstractLookup.Pair[] arr = new AbstractLookup.Pair[items.size()]; Iterator it = items.iterator(); for (int i = 0; i < arr.length; i++) { arr[i] = new PairWrap(it.next()); } QUERY_MODE.set(arr); } /** Add a listener to changes in the cookie set. * @param l the listener to add */ public void addChangeListener(ChangeListener l) { cs.addChangeListener(l); } /** Remove a listener to changes in the cookie set. * @param l the listener to remove */ public void removeChangeListener(ChangeListener l) { cs.removeChangeListener(l); } /** Node lookup starts its non-important query. */ static Object entryQueryMode(Class c) { Object prev = QUERY_MODE.get(); QUERY_MODE.set(c); return prev; } /** Allows query for all know classes registered in this cookie. */ static Object entryAllClassesMode() { Object prev = QUERY_MODE.get(); QUERY_MODE.set(new HashSet()); return prev; } /** Exits query mode. */ static Collection exitQueryMode(Object prev) { Object cookie = QUERY_MODE.get(); QUERY_MODE.set(prev); if (cookie instanceof CookieSet.CookieEntry) { return Collections.singleton((AbstractLookup.Pair)new CookieEntryPair((CookieSet.CookieEntry) cookie)); } else if (cookie instanceof AbstractLookup.Pair[]) { return Arrays.asList((AbstractLookup.Pair[])cookie); } else { return null; } } /** Returns list of all classes. */ static Set exitAllClassesMode(Object prev) { Object cookie = QUERY_MODE.get(); QUERY_MODE.set(prev); if (cookie instanceof HashSet) { return (Set) cookie; } return null; } /** Fires change event */ final void fireChangeEvent() { cs.fireChange(); } /** Attaches cookie to given class and all its superclasses and * superinterfaces. * * @param c class or null * @param cookie cookie to attach */ private void registerCookie(Class c, Object cookie) { if ((c == null) || !Node.Cookie.class.isAssignableFrom(c)) { return; } Class nc = c.asSubclass(Node.Cookie.class); R r = findR(nc); if (r == null) { r = new R(); map.put(c, r); } r.add((Node.Cookie)cookie); registerCookie(c.getSuperclass(), cookie); Class[] inter = c.getInterfaces(); for (int i = 0; i < inter.length; i++) { registerCookie(inter[i], cookie); } } /** Removes cookie from the class and all its superclasses and * superinterfaces. * * @param c class or null * @param cookie cookie to attach */ private void unregisterCookie(Class c, Object cookie) { if ((c == null) || !Node.Cookie.class.isAssignableFrom(c)) { return; } Class nc = c.asSubclass(Node.Cookie.class); // if different cookie is attached to class c stop removing R r = findR(nc); if (r != null) { // remove the cookie r.remove((Node.Cookie)cookie); } unregisterCookie(c.getSuperclass(), cookie); Class[] inter = c.getInterfaces(); for (int i = 0; i < inter.length; i++) { unregisterCookie(inter[i], cookie); } } /** Registers a Factory for given cookie class */ public void add(Class cookieClass, Factory factory) { if (factory == null) { throw new IllegalArgumentException(); } final CookieEntry ce = new CookieEntry(factory, cookieClass); synchronized (this) { registerCookie(cookieClass, ce); } if (ic != null) { ic.add(ce, C.INSTANCE); } fireChangeEvent(); } /** Registers a Factory for given cookie classes */ public void add(Class[] cookieClass, Factory factory) { if (factory == null) { throw new IllegalArgumentException(); } CookieEntry[] arr = new CookieEntry[cookieClass.length]; synchronized (this) { for (int i = 0; i < cookieClass.length; i++) { registerCookie(cookieClass[i], arr[i] = new CookieEntry(factory, cookieClass[i])); } } if (ic != null) { int i = 0; for (Class c : cookieClass) { ic.add(arr[i++], C.INSTANCE); } } fireChangeEvent(); } /** * Unregisters a Factory for given cookie class * @since 2.6 */ public void remove(Class cookieClass, Factory factory) { if (factory == null) { throw new IllegalArgumentException(); } CookieEntry ce = null; synchronized (this) { R r = findR(cookieClass); if (r != null) { Node.Cookie c = r.cookie(); if (c instanceof CookieEntry) { ce = (CookieEntry) c; if (ce.factory == factory) { unregisterCookie(cookieClass, c); } } } } if (ic != null && ce != null) { ic.remove(ce, C.INSTANCE); } fireChangeEvent(); } /** * Unregisters a Factory for given cookie classes * @since 2.6 */ public void remove(Class[] cookieClass, Factory factory) { if (factory == null) { throw new IllegalArgumentException(); } CookieEntry[] arr = new CookieEntry[cookieClass.length]; synchronized (this) { for (int i = 0; i < cookieClass.length; i++) { R r = findR(cookieClass[i]); if (r != null) { Node.Cookie c = r.cookie(); if (c instanceof CookieEntry) { CookieEntry ce = (CookieEntry) c; arr[i] = ce; if (ce.factory == factory) { unregisterCookie(cookieClass[i], c); } } } } } if (ic != null) { for (CookieEntry ce : arr) { if (ce != null) { ic.remove(ce, C.INSTANCE); } } } fireChangeEvent(); } /** Removes all instances of clazz from the set and replaces them * with newly provided instance(s). * * @param clazz the root clazz for cookies to remove * @param instances the one or more instances to put into the lookup * * @since 7.0 */ public void assign(Class clazz, T... instances) { if (Node.Cookie.class.isAssignableFrom(clazz)) { Class cookieClazz = clazz.asSubclass(Node.Cookie.class); for(;;) { Node.Cookie cookie = lookupCookie(cookieClazz); if (cookie != null) { synchronized (this) { R r = findR(cookie.getClass()); if (r != null && r.cookies != null) { List arr = r.cookies; r.base = null; r.cookies = null; for (Cookie c : arr) { unregisterCookie(cookie.getClass(), c); } } } unregisterCookie(cookie.getClass(), cookie); if (ic != null) { ic.remove(cookie); } } else { break; } } for (T t : instances) { addImpl(t); } fireChangeEvent(); } else if (ic != null) { synchronized (this) { for (T t : instances) { registerCookie(t.getClass(), t); } } ic.replaceInstances(clazz, instances, this); } } /** Assignes a trigger that gets called everytime given class is about * to be queried. Can be used only for cookie set created with * {@link CookieSet#create(true)} and for classes that are not * subclasses of Node.Cookie. * * @param clazz the trigger class (not subclass of Node.Cookie) * @param run runnable to run when the task is queried * public void beforeLookup(Class clazz, Runnable run) { if (Node.Cookie.class.isAssignableFrom(clazz) || clazz == Object.class) { throw new IllegalArgumentException("Too generic class: " + clazz); // NOI18N } if (ic == null) { throw new IllegalStateException("Can be used only on CookieSet.create(true)"); // NOI18N } ic.registerBeforeLookup(clazz, run); } */ /** Finds a result in a map. */ private R findR(Class c) { return map.get(c); } /** Finds base class for a cookie. * @param c cookie * @return base class */ private static Class baseForCookie(Node.Cookie c) { if (c instanceof CookieEntry) { return ((CookieEntry) c).klass; } return c.getClass(); } /** Factory for creating cookies of given Class */ public interface Factory { /** Creates a Node.Cookie of given class. The method * may be called more than once. */ T createCookie(Class klass); } /** Allows to update content of the cookie set just before * a query for a given class is made. */ public interface Before { public void beforeLookup(Class clazz); } /** Entry for one Cookie */ private static class CookieEntry implements Node.Cookie { /** Factory for the cookie */ final Factory factory; /** Class of the cookie */ private final Class klass; private Reference cookie; /** Constructs new FactoryEntry */ public CookieEntry(Factory factory, Class klass) { this.factory = factory; this.klass = klass; } /** Getter for the cookie. * Synchronized because we don't want to run factory.createCookie * simultaneously from two threads. */ public synchronized Node.Cookie getCookie(boolean create) { Node.Cookie ret; if (create) { if ((cookie == null) || ((ret = cookie.get()) == null)) { ret = factory.createCookie(klass); if (ret == null) { return null; } cookie = new WeakReference(ret); } } else { ret = (cookie == null) ? null : cookie.get(); } return ret; } } // end of CookieEntry /** Implementation of the result. */ private static final class R extends Object { /** list of registered cookies */ public List cookies; /** base class of the first cookie registered here */ public Class base; R() { } /** Adds a cookie. * @return true if adding should continue on superclasses should continue */ public void add(Node.Cookie cookie) { if (cookies == null) { cookies = new ArrayList(1); cookies.add(cookie); base = baseForCookie(cookie); return; } Class newBase = baseForCookie(cookie); if ((base == null) || newBase.isAssignableFrom(base)) { cookies.set(0, cookie); base = newBase; } else { cookies.add(cookie); } } /** Removes a cookie. * @return true if empty */ public boolean remove(Node.Cookie cookie) { if (cookies == null) { return true; } if (cookies.remove(cookie) && (cookies.size() == 0)) { base = null; cookies = null; return true; } base = baseForCookie(cookies.get(0)); return false; } /** @return the cookie for this result or null */ public Node.Cookie cookie() { return ((cookies == null) || cookies.isEmpty()) ? null : cookies.get(0); } } /** Pair that wraps another Lookup.Item */ static final class PairWrap extends AbstractLookup.Pair { private Lookup.Item item; private boolean created; public PairWrap(Lookup.Item item) { this.item = item; } protected boolean instanceOf(Class c) { Class k = c; return k.isAssignableFrom(getType()); } protected boolean creatorOf(Object obj) { return created && getInstance() == obj; } public Object getInstance() { created = true; return item.getInstance(); } public Class getType() { return item.getType(); } public String getId() { return item.getId(); } public String getDisplayName() { return item.getDisplayName(); } @Override public int hashCode() { return 777 + item.hashCode(); } @Override public boolean equals(Object object) { if (object instanceof PairWrap) { PairWrap p = (PairWrap)object; return item.equals(p.item); } return false; } } // end of PairWrap /** Pair that represents an entry. */ private static final class CookieEntryPair extends AbstractLookup.Pair { private CookieEntry entry; public CookieEntryPair(CookieEntry e) { this.entry = e; } protected boolean creatorOf(Object obj) { return obj == entry.getCookie(false); } public String getDisplayName() { return getId(); } public String getId() { return entry.klass.getName(); } public Object getInstance() { return entry.getCookie(true); } public Class getType() { return entry.klass; } protected boolean instanceOf(Class c) { Class k = c; return k.isAssignableFrom(entry.klass); } @Override public int hashCode() { return entry.hashCode() + 5; } @Override public boolean equals(Object obj) { if (obj instanceof CookieEntryPair) { return ((CookieEntryPair) obj).entry == entry; } return false; } } // end of CookieEntryPair static class C implements InstanceContent.Convertor { static final C INSTANCE = new C(); public Node.Cookie convert(CookieSet.CookieEntry obj) { return obj.getCookie(true); } public Class type(CookieSet.CookieEntry obj) { return obj.klass; } public String id(CookieSet.CookieEntry obj) { return obj.klass.getName(); } public String displayName(CookieSet.CookieEntry obj) { return obj.klass.getName(); } } }