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

org.openide.nodes.EntrySupportDefault 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.AbstractList;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.openide.nodes.Children.Entry;
import org.openide.util.Utilities;

/** Default support that just fires changes directly to children and is suitable
 * for simple mappings.
 */
class EntrySupportDefault extends EntrySupport {
    private List entries = Collections.emptyList();

    private static final Reference EMPTY = new WeakReference(null);
    /** array of children Reference (ChildrenArray) */
    private Reference array = EMPTY;
    /** mapping from entries to info about them */
    private Map map;
    private static final Object LOCK = new Object();
    private static final Logger LOGGER = Logger.getLogger(EntrySupportDefault.class.getName()); // NOI18N
    //private static final boolean LOG_ENABLED = LOGGER.isLoggable(Level.FINER);
    private Thread initThread;
    private boolean inited = false;

    public EntrySupportDefault(Children ch) {
        super(ch);
    }

    @Override
    public String toString() {
        return super.toString() + " array: " + array.get(); // NOI18N
    }
    
    

    public boolean isInitialized() {
        ChildrenArray arr = array.get();
        return inited && arr != null && arr.isInitialized();
    }

    @Override
    List snapshot() {
        Node[] nodes = getNodes();
        try {
            Children.PR.enterReadAccess();
            return createSnapshot();
        } finally {
            Children.PR.exitReadAccess();
        }
    }

    DefaultSnapshot createSnapshot() {
        return new DefaultSnapshot(getNodes(), array.get());
    }

    public final Node[] getNodes() {
        final boolean LOG_ENABLED = LOGGER.isLoggable(Level.FINER);
        if (LOG_ENABLED) {
            LOGGER.finer("getNodes() " + this);
        }
        boolean[] results = new boolean[2];
        for (;;) {
            // initializes the ChildrenArray possibly calls
            // addNotify if this is for the first time
            ChildrenArray tmpArray = getArray(results); // fils results[0]
            Node[] nodes;
            try {
                Children.PR.enterReadAccess();
                if (this != children.getEntrySupport()) {
                    // support was switched while we were waiting for access
                    return new Node[0];
                }
                results[1] = isInitialized();
                nodes = tmpArray.nodes();
            } finally {
                Children.PR.exitReadAccess();
            }
            if (LOG_ENABLED) {
                LOGGER.finer("  length     : " + (nodes == null ? "nodes is null" : nodes.length)); // NOI18N
                LOGGER.finer("  init now   : " + isInitialized()); // NOI18N
            }
            // if not initialized that means that after
            // we computed the nodes, somebody changed them (as a
            // result of addNotify) => we have to compute them
            // again
            if (results[1]) {
                // otherwise it is ok.
                return nodes;
            }
            if (results[0]) {
                // looks like the result cannot be computed, just give empty one
                notifySetEntries();
                return (nodes == null) ? new Node[0] : nodes;
            }
        }
    }

    public Node[] getNodes(boolean optimalResult) {
        final boolean LOG_ENABLED = LOGGER.isLoggable(Level.FINER);
        ChildrenArray hold = null;
        Node find = null;
        if (optimalResult) {
            if (LOG_ENABLED) {
                LOGGER.finer("computing optimal result"); // NOI18N
            }
            hold = getArray(null);
            if (LOG_ENABLED) {
                LOGGER.finer("optimal result is here: " + hold); // NOI18N
            }
            find = children.findChild(null);
            if (LOG_ENABLED) {
                LOGGER.finer("Find child got: " + find); // NOI18N
            }
            Children.LOG.log(Level.FINEST, "after findChild: {0}", optimalResult);
        }
        return getNodes();
    }

    public final int getNodesCount(boolean optimalResult) {
        return getNodes(optimalResult).length;
    }

    @Override
    public Node getNodeAt(int index) {
        Node[] nodes = getNodes();
        return index < nodes.length ? nodes[index] : null;
    }

    /** Computes the nodes now.
     */
    final Node[] justComputeNodes() {
        if (map == null) {
            map = Collections.synchronizedMap(new HashMap(17));
            LOGGER.finer("Map initialized");
        }
        List l = new LinkedList();
        for (Entry entry : entries) {
            Info info = findInfo(entry);
            l.addAll(info.nodes(false));
        }
        Node[] arr = l.toArray(new Node[l.size()]);
        // initialize parent nodes
        for (int i = 0; i < arr.length; i++) {
            Node n = arr[i];
            if (n == null) {
                LOGGER.warning("null node among children!");
                for (int j = 0; j < arr.length; j++) {
                    LOGGER.log(Level.WARNING, "  {0} = {1}", new Object[]{j, arr[j]});
                }
                for (Entry entry : entries) {
                    Info info = findInfo(entry);
                    LOGGER.log(Level.WARNING, "  entry: {0} info {1} nodes: {2}", new Object[]{entry, info, info.nodes(false)});
                }
                throw new NullPointerException("arr[" + i + "] is null"); // NOI18N
            }
            n.assignTo(children, i);
            n.fireParentNodeChange(null, children.parent);
        }
        return arr;
    }

    /** Finds info for given entry, or registers
     * it, if not registered yet.
     */
    private Info findInfo(Entry entry) {
        synchronized (map) {
            Info info = map.get(entry);
            if (info == null) {
                info = new Info(entry);
                map.put(entry, info);
                if (LOGGER.isLoggable(Level.FINER)) {
                    LOGGER.finer("Put: " + entry + " info: " + info);
                }
            }
            return info;
        }
    }
    //
    // Entries
    //
    private boolean mustNotifySetEnties = false;

    void notifySetEntries() {
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer(this + " mustNotifySetEntries()");
        }
        mustNotifySetEnties = true;
    }

    private void checkConsistency() {
        assert map.size() == this.entries.size() : "map.size()=" + map.size() + " entries.size()=" + this.entries.size();
    }

    @Override
    protected void setEntries(Collection entries, boolean noCheck) {
        assert noCheck || Children.MUTEX.isWriteAccess();
        final boolean LOG_ENABLED = LOGGER.isLoggable(Level.FINER);
        // current list of nodes
        ChildrenArray holder = array.get();
        if (LOG_ENABLED) {
            LOGGER.finer("setEntries for " + this + " on " + Thread.currentThread()); // NOI18N
            LOGGER.finer("       values: " + entries); // NOI18N
            LOGGER.finer("       holder: " + holder); // NOI18N
            LOGGER.finer("       mustNotifySetEntries: " + mustNotifySetEnties); // NOI18N
        }
        Node[] current = holder == null ? null : holder.nodes();
        if (mustNotifySetEnties) {
            if (holder == null) {
                holder = getArray(null);
            }
            if (current == null) {
                holder.entrySupport = this;
                current = holder.nodes();
            }
            mustNotifySetEnties = false;
        } else if (holder == null || current == null) {
            this.entries = new ArrayList(entries);
            if (map != null) {
                map.keySet().retainAll(new HashSet(this.entries));
            }
            return;
        }
        checkConsistency();
        // what should be removed
        Set toRemove = new LinkedHashSet(this.entries);
        Set entriesSet = new HashSet(entries);
        toRemove.removeAll(entriesSet);
        if (!toRemove.isEmpty()) {
            // notify removing, the set must be ready for
            // callbacks with questions
            updateRemove(current, toRemove);
            current = holder.nodes();
        }
        // change the order of entries, notifies
        // it and again brings children to up-to-date state
        Collection toAdd = updateOrder(current, entries);
        if (!toAdd.isEmpty()) {
            // toAdd contains Info objects that should bee added
            updateAdd(toAdd, new ArrayList(entries));
        }
    }

    private void checkInfo(Info info, Entry entry, Collection entries, java.util.Map map) {
        if (info == null) {
            StringBuilder sb = new StringBuilder();
            sb.append("Error in ").append(getClass().getName()).
                append(" with entry ").append(entry).append(" from among entries:");
            for (Entry e : entries) {
                sb.append("\n  ").append(e).append(" contained: ").append(map.containsKey(e));
            }
            sb.append("\nprobably caused by faulty key implementation. The key hashCode() and equals() methods must behave as for an IMMUTABLE object" + " and the hashCode() must return the same value for equals() keys."); // NOI18N
            sb.append("\nmapping:");
            for (Map.Entry ei : map.entrySet()) {
                sb.append("\n  ").append(ei.getKey()).append(" => ").append(ei.getValue());
            }
            throw new IllegalStateException(sb.toString());
        }
    }

    /** Removes the objects from the children.
     */
    private void updateRemove(Node[] current, Set toRemove) {
        assert Children.MUTEX.isWriteAccess();
        List nodes = new LinkedList();
        ChildrenArray cha = array.get();
        for (Entry en : toRemove) {
            Info info = map.remove(en);
            checkInfo(info, en, new ArrayList(), map);
            nodes.addAll(info.nodes(true));
            cha.remove(info);
        }
        // modify the current set of entries
        entries.removeAll(toRemove);
        checkConsistency();
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer("Current : " + this.entries);
            LOGGER.finer("Removing: " + toRemove);
        }
        // empty the list of nodes so it has to be recreated again
        if (!nodes.isEmpty()) {
            clearNodes();
            notifyRemove(nodes, current);
        }
    }

    /** Updates the order of entries.
     * @param current current state of nodes
     * @param entries new set of entries
     * @return list of infos that should be added
     */
    private List updateOrder(Node[] current, Collection newEntries) {
        assert Children.MUTEX.isWriteAccess();
        List toAdd = new LinkedList();
        // that assignes entries their begining position in the array
        // of nodes
        java.util.Map offsets = new HashMap();
        {
            int previousPos = 0;
            for (Entry entry : entries) {
                Info info = map.get(entry);
                checkInfo(info, entry, entries, map);
                offsets.put(info, previousPos);
                previousPos += info.length();
            }
        }
        int[] perm = new int[current.length];
        int currentPos = 0;
        int permSize = 0;
        List reorderedEntries = null;
        for (Entry entry : newEntries) {
            Info info = map.get(entry);
            if (info == null) {
                // this info has to be added
                info = new Info(entry);
                toAdd.add(info);
            } else {
                int len = info.length();
                if (reorderedEntries == null) {
                    reorderedEntries = new LinkedList();
                }
                reorderedEntries.add(entry);
                // already there => test if it should not be reordered
                Integer previousInt = offsets.get(info);
                /*
                if (previousInt == null) {
                System.err.println("Offsets: " + offsets);
                System.err.println("Info: " + info);
                System.err.println("Entry: " + info.entry);
                System.err.println("This entries: " + this.entries);
                System.err.println("Entries: " + entries);
                System.err.println("Map: " + map);
                System.err.println("---------vvvvv");
                System.err.println(debug);
                System.err.println("---------^^^^^");
                }
                 */
                int previousPos = previousInt;
                if (currentPos != previousPos) {
                    for (int i = 0; i < len; i++) {
                        perm[previousPos + i] = 1 + currentPos + i;
                    }
                    permSize += len;
                }
            }
            currentPos += info.length();
        }
        if (permSize > 0) {
            // now the perm array contains numbers 1 to ... and
            // 0 one places where no permutation occures =>
            // decrease numbers, replace zeros
            for (int i = 0; i < perm.length; i++) {
                if (perm[i] == 0) {
                    // fixed point
                    perm[i] = i;
                } else {
                    // decrease
                    perm[i]--;
                }
            }
            if (LOGGER.isLoggable(Level.FINER)) {
                LOGGER.finer("Entries before reordering: " + entries);
                LOGGER.finer("Entries after reordering: " + reorderedEntries);
            }
            // reorderedEntries are not null
            entries = reorderedEntries;
            checkConsistency();
            // notify the permutation to the parent
            clearNodes();
            Node p = children.parent;
            if (p != null) {
                p.fireReorderChange(perm);
            }
        }
        return toAdd;
    }

    /** Updates the state of children by adding given Infos.
     * @param infos list of Info objects to add
     * @param entries the final state of entries that should occur
     */
    private void updateAdd(Collection infos, List entries) {
        assert Children.MUTEX.isWriteAccess();
        List nodes = new LinkedList();
        for (Info info : infos) {
            nodes.addAll(info.nodes(false));
            map.put(info.entry, info);
        }
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer("Entries before updateAdd(): " + this.entries);
            LOGGER.finer("Entries after updateAdd(): " + entries);
        }
        this.entries = entries;
        checkConsistency();
        if (!nodes.isEmpty()) {
            clearNodes();
            notifyAdd(nodes);
        }
    }

    /** Refreshes content of one entry. Updates the state of children
     * appropriately.
     */
    final void refreshEntry(Entry entry) {
        // current list of nodes
        ChildrenArray holder = array.get();
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer("refreshEntry: " + entry + " holder=" + holder);
        }
        if (holder == null) {
            return;
        }
        Node[] current = holder.nodes();
        if (current == null) {
            // the initialization is not finished yet =>
            return;
        }
        checkConsistency();
        Info info = map.get(entry);
        if (info == null) {
            // refresh of entry that is not present =>
            return;
        }
        Collection oldNodes = info.nodes(false);
        Collection newNodes = info.entry.nodes(null);
        if (oldNodes.equals(newNodes)) {
            // nodes are the same =>
            return;
        }
        Set toRemove = new HashSet(oldNodes);
        toRemove.removeAll(new HashSet(newNodes));
        if (!toRemove.isEmpty()) {
            // notify removing, the set must be ready for
            // callbacks with questions
            // modifies the list associated with the info
            oldNodes.removeAll(toRemove);
            clearNodes();
            // now everything should be consistent => notify the remove
            notifyRemove(toRemove, current);
            current = holder.nodes();
        }
        List toAdd = refreshOrder(entry, oldNodes, newNodes);
        info.useNodes(newNodes);
        if (!toAdd.isEmpty()) {
            // modifies the list associated with the info
            clearNodes();
            notifyAdd(toAdd);
        }
    }

    /** Updates the order of nodes after a refresh.
     * @param entry the refreshed entry
     * @param oldNodes nodes that are currently in the list
     * @param newNodes new nodes (defining the order of oldNodes and some more)
     * @return list of infos that should be added
     */
    private List refreshOrder(Entry entry, Collection oldNodes, Collection newNodes) {
        List toAdd = new LinkedList();
        Set oldNodesSet = new HashSet(oldNodes);
        Set toProcess = new HashSet(oldNodesSet);
        Node[] permArray = new Node[oldNodes.size()];
        Iterator it2 = newNodes.iterator();
        int pos = 0;
        while (it2.hasNext()) {
            Node n = it2.next();
            if (oldNodesSet.remove(n)) {
                // the node is in the old set => test for permuation
                permArray[pos++] = n;
            } else {
                if (!toProcess.contains(n)) {
                    // if the node has not been processed yet
                    toAdd.add(n);
                } else {
                    it2.remove();
                }
            }
        }
        // JST: If you get IllegalArgumentException in following code
        // then it can be cause by wrong synchronization between
        // equals and hashCode methods. First of all check them!
        int[] perm = NodeOp.computePermutation(oldNodes.toArray(new Node[oldNodes.size()]), permArray);
        if (perm != null) {
            // apply the permutation
            clearNodes();
            // temporarily change the nodes the entry should use
            findInfo(entry).useNodes(Arrays.asList(permArray));
            Node p = children.parent;
            if (p != null) {
                p.fireReorderChange(perm);
            }
        }
        return toAdd;
    }

    /** Notifies that a set of nodes has been removed from
     * children. It is necessary that the system is already
     * in consistent state, so any callbacks will return
     * valid values.
     *
     * @param nodes list of removed nodes
     * @param current state of nodes
     * @return array of nodes that were deleted
     */
    Node[] notifyRemove(Collection nodes, Node[] current) {
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer("notifyRemove: " + nodes);
            LOGGER.finer("Current     : " + Arrays.asList(current));
        }
        // during a deserialization it may have parent == null
        Node[] arr = nodes.toArray(new Node[nodes.size()]);
        if (children.parent != null) {
            // fire change of nodes
            if (children.getEntrySupport() == this) {
                children.parent.fireSubNodesChange(false, arr, current);
            }
            // fire change of parent
            Iterator it = nodes.iterator();
            while (it.hasNext()) {
                Node n = it.next();
                n.deassignFrom(children);
                n.fireParentNodeChange(children.parent, null);
            }
        }
        children.destroyNodes(arr);
        return arr;
    }

    /** Notifies that a set of nodes has been add to
     * children. It is necessary that the system is already
     * in consistent state, so any callbacks will return
     * valid values.
     *
     * @param nodes list of removed nodes
     */
    void notifyAdd(Collection nodes) {
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer("notifyAdd: " + nodes);
        }
        // notify about parent change
        for (Node n : nodes) {
            n.assignTo(children, -1);
            n.fireParentNodeChange(null, children.parent);
        }
        Node[] arr = nodes.toArray(new Node[nodes.size()]);
        Node n = children.parent;
        if (n != null && children.getEntrySupport() == this) {
            n.fireSubNodesChange(true, arr, null);
        }
    }

    /**
     * @return either nodes associated with this children or null if they are not created
     */
    public Node[] testNodes() {
        ChildrenArray arr = array.get();
        if (arr == null) {
            return null;
        }
        try {
            Children.PR.enterReadAccess();
            return arr.nodes();
        } finally {
            Children.PR.exitReadAccess();
        }
    }

    /** Obtains references to array holder. If it does not exist, it is created.
     *
     * @param cannotWorkBetter array of size 1 or null, will contain true, if
     *    the getArray cannot be initialized (we are under read access
     *    and another thread is responsible for initialization, in such case
     *    give up on computation of best result
     */
    private ChildrenArray getArray(boolean[] cannotWorkBetter) {
        final boolean LOG_ENABLED = LOGGER.isLoggable(Level.FINER);
        ChildrenArray arr;
        boolean doInitialize = false;
        synchronized (LOCK) {
            arr = array.get();
            if (arr == null) {
                arr = new ChildrenArray();
                // register the array with the children
                registerChildrenArray(arr, true);
                doInitialize = true;
                initThread = Thread.currentThread();
            }
        }
        if (doInitialize) {
            if (LOG_ENABLED) {
                LOGGER.finer("Initialize " + this + " on " + Thread.currentThread()); // NOI18N
            }
            // this call can cause a lot of callbacks => be prepared
            // to handle them as clean as possible
            try {
                children.callAddNotify();
                if (LOG_ENABLED) {
                    LOGGER.finer("addNotify successfully called for " + this + " on " + Thread.currentThread()); // NOI18N
                }
            } finally {
                boolean notifyLater;
                notifyLater = Children.MUTEX.isReadAccess();
                if (LOG_ENABLED) {
                    LOGGER.finer("notifyAll for " + this + " on " + Thread.currentThread() + "  notifyLater: " + notifyLater); // NOI18N
                }
                // now attach to entrySupport, so when entrySupport == null => we are
                // not fully initialized!!!!
                arr.entrySupport = this;
                inited = true;

                class SetAndNotify implements Runnable {

                    public ChildrenArray toSet;
                    public Children whatSet;

                    public void run() {
                        synchronized (LOCK) {
                            initThread = null;
                            LOCK.notifyAll();
                        }
                        if (LOG_ENABLED) {
                            LOGGER.finer("notifyAll done"); // NOI18N
                        }
                    }
                }
                SetAndNotify setAndNotify = new SetAndNotify();
                setAndNotify.toSet = arr;
                setAndNotify.whatSet = children;
                if (notifyLater) {
                    // the notify to the lock has to be done later than
                    // setKeys is executed, otherwise the result of addNotify
                    // might not be visible to other threads
                    // fix for issue 50308
                    Children.MUTEX.postWriteRequest(setAndNotify);
                } else {
                    setAndNotify.run();
                }
            }
        } else if (initThread != null) {
            // otherwise, if not initialize yet (arr.children) wait
            // for the initialization to finish, but only if we can wait
            if (Children.MUTEX.isReadAccess() || Children.MUTEX.isWriteAccess() || (initThread == Thread.currentThread())) {
                // fail, we are in read access
                if (LOG_ENABLED) {
                    LOGGER.log(Level.FINER, "cannot initialize better " + this + " on " + Thread.currentThread() + " read access: " + Children.MUTEX.isReadAccess() + " write access: " + Children.MUTEX.isWriteAccess() + " initThread: " + initThread);
                }
                if (cannotWorkBetter != null) {
                    cannotWorkBetter[0] = true;
                }
                arr.entrySupport = this;
                return arr;
            }
            // otherwise we can wait
            synchronized (LOCK) {
                while (initThread != null) {
                    if (LOG_ENABLED) {
                        LOGGER.finer("waiting for children for " + this + " on " + Thread.currentThread());
                    }
                    try {
                        LOCK.wait();
                    } catch (InterruptedException ex) {
                    }
                }
            }
            if (LOG_ENABLED) {
                LOGGER.finer(" children are here for " + this + " on " + Thread.currentThread() + " children " + children); // NOI18N
            }
        }
        return arr;
    }

    /** Clears the nodes
     */
    private void clearNodes() {
        if (LOGGER.isLoggable(Level.FINER)) {
            LOGGER.finer("  clearNodes()"); // NOI18N
        }
        ChildrenArray arr = array.get();
        if (arr != null) {
            // clear the array
            arr.clear();
        }
    }

    /** Registration of ChildrenArray.
     * @param chArr the associated ChildrenArray
     * @param weak use weak or hard reference
     */
    final void registerChildrenArray(final ChildrenArray chArr, boolean weak) {
        final boolean LOG_ENABLED = LOGGER.isLoggable(Level.FINER);
        if (LOG_ENABLED) {
            LOGGER.finer("registerChildrenArray: " + chArr + " weak: " + weak); // NOI18N
        }
        synchronized (LOCK) {
            if (this.array != null && this.array.get() == chArr && ((ChArrRef) this.array).isWeak() == weak) {
                return;
            }
            this.array = new ChArrRef(chArr, weak);
        }
        if (LOG_ENABLED) {
            LOGGER.finer("pointed by: " + chArr + " to: " + this.array); // NOI18N
        }
    }

    /** Finalized.
     */
    final void finalizedChildrenArray(Reference caller) {
        assert caller.get() == null : "Should be null";
        // usually in removeNotify setKeys is called => better require write access
        try {
            Children.PR.enterWriteAccess();
            if (LOGGER.isLoggable(Level.FINER)) {
                LOGGER.fine("previous array: " + array + " caller: " + caller); // NOI18N
            }
            synchronized (LOCK) {
                if (array == caller && children.getEntrySupport() == this) {
                    // really finalized and not reconstructed
                    mustNotifySetEnties = false;
                    array = EMPTY;
                    inited = false;
                    children.callRemoveNotify();
                    assert array == EMPTY;
                }
            }
        } finally {
            Children.PR.exitWriteAccess();
        }
    }

    @Override
    protected List getEntries() {
        return new ArrayList(entries);
    }

    /** Information about an entry. Contains number of nodes,
     * position in the array of nodes, etc.
     */
    final class Info extends Object {

        int length;
        final Entry entry;

        public Info(Entry entry) {
            this.entry = entry;
        }

        public Collection nodes(boolean hasToExist) {
            // forces creation of the array
            assert !hasToExist || array.get() != null : "ChildrenArray is not initialized";
            ChildrenArray arr = getArray(null);
            return arr.nodesFor(this, hasToExist);
        }

        public void useNodes(Collection nodes) {
            // forces creation of the array
            ChildrenArray arr = getArray(null);
            arr.useNodes(this, nodes);
            // assign all there nodes the new children
            for (Node n : nodes) {
                n.assignTo(EntrySupportDefault.this.children, -1);
                n.fireParentNodeChange(null, children.parent);
            }
        }

        public int length() {
            return length;
        }

        @Override
        public String toString() {
            return "Children.Info[" + entry + ",length=" + length + "]"; // NOI18N
        }
    }

    static class DefaultSnapshot extends AbstractList {

        private Node[] nodes;
        Object holder;

        public DefaultSnapshot(Node[] nodes, ChildrenArray cha) {
            this.nodes = nodes;
            this.holder = cha;
        }

        public Node get(int index) {
            return nodes != null && index < nodes.length ? nodes[index] : null;
        }

        public int size() {
            return nodes != null ? nodes.length : 0;
        }
    }

    private class ChArrRef extends WeakReference implements Runnable {
        private final ChildrenArray chArr;

        public ChArrRef(ChildrenArray referent, boolean weak) {
            super(referent, Utilities.activeReferenceQueue());
            this.chArr = weak ? null : referent;
        }

        @Override
        public ChildrenArray get() {
            return chArr != null ? chArr : super.get();
        }

        boolean isWeak() {
            return chArr == null;
        }

        @Override
        public void run() {
            finalizedChildrenArray(this);
        }
    }
    
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy