/*
 * Copyright (c) 2010, 2017, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package com.sun.javafx.scene.control;

import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import java.util.NoSuchElementException;

import javafx.beans.InvalidationListener;
import com.sun.javafx.collections.ListListenerHelper;
import javafx.collections.ListChangeListener;
import javafx.collections.ListChangeListener.Change;
import javafx.collections.ObservableList;
import javafx.collections.ObservableListBase;

import java.util.Collections;
import java.util.function.Consumer;

import static com.sun.javafx.collections.ListListenerHelper.fireValueChangedEvent;

A read-only and unbacked ObservableList - the data is retrieved on demand by subclasses via the get method. A combination of ObservableList, ObservableListWrapper and GenericObservableList.
/** * A read-only and unbacked ObservableList - the data is retrieved on demand by * subclasses via the get method. A combination of ObservableList, ObservableListWrapper * and GenericObservableList. * */
public abstract class ReadOnlyUnbackedObservableList<E> extends ObservableListBase<E> { @Override public abstract E get(int i); @Override public abstract int size(); public void _beginChange() { beginChange(); } public void _endChange() { endChange(); } public void _nextUpdate(int pos) { nextUpdate(pos); } public void _nextSet(int idx, E old) { nextSet(idx, old); } public void _nextReplace(int from, int to, List<? extends E> removed) { nextReplace(from, to, removed); } public void _nextRemove(int idx, List<? extends E> removed) { nextRemove(idx, removed); } public void _nextRemove(E o) { int indexOfObject = indexOf(o); _nextRemove(indexOfObject, o); } public void _nextRemove(int idx, E removed) { nextRemove(idx, removed); } public void _nextPermutation(int from, int to, int[] perm) { nextPermutation(from, to, perm); } public void _nextAdd(int from, int to) { nextAdd(from, to); } public void fireChange(Runnable r) { _beginChange(); r.run(); _endChange(); } public void callObservers(Change<E> c) { fireChange(c); } @Override public int indexOf(Object o) { if (o == null) return -1; for (int i = 0; i < size(); i++) { Object obj = get(i); if (o.equals(obj)) return i; } return -1; } @Override public int lastIndexOf(Object o) { if (o == null) return -1; for (int i = size() - 1; i >= 0; i--) { Object obj = get(i); if (o.equals(obj)) return i; } return -1; } @Override public boolean contains(Object o) { return indexOf(o) != -1; } @Override public boolean containsAll(Collection<?> c) { for (Object o : c) { if (! contains(o)) { return false; } } return true; } @Override public boolean isEmpty() { return size() == 0; } @Override public ListIterator<E> listIterator() { return new SelectionListIterator<E>(this); } @Override public ListIterator<E> listIterator(int index) { return new SelectionListIterator<E>(this, index); } @Override public Iterator<E> iterator() { return new SelectionListIterator<E>(this); }
NOTE: This method does not fulfill the subList contract from Collections, it simply returns a list containing the values in the given range.
/** * NOTE: This method does not fulfill the subList contract from Collections, * it simply returns a list containing the values in the given range. */
@Override public List<E> subList(final int fromIndex, final int toIndex) { if (fromIndex < 0 || toIndex > size() || fromIndex > toIndex) { throw new IndexOutOfBoundsException("[ fromIndex: " + fromIndex + ", toIndex: " + toIndex + ", size: " + size() + " ]"); } final List<E> outer = this; return new ReadOnlyUnbackedObservableList<E>() { @Override public E get(int i) { return outer.get(i + fromIndex); } @Override public int size() { return toIndex - fromIndex; } }; } @Override public Object[] toArray() { Object[] arr = new Object[size()]; for (int i = 0; i < size(); i++) { arr[i] = get(i); } return arr; } @SuppressWarnings("unchecked") @Override public <T> T[] toArray(T[] a) { Object[] elementData = toArray(); int size = elementData.length; if (a.length < size) // Make a new array of a's runtime type, but my contents: return (T[]) Arrays.copyOf(elementData, size, a.getClass()); System.arraycopy(elementData, 0, a, 0, size); if (a.length > size) a[size] = null; return a; } @Override public String toString() { // copied from AbstractCollection Iterator<E> i = iterator(); if (! i.hasNext()) return "[]"; StringBuilder sb = new StringBuilder(); sb.append('['); for (;;) { E e = i.next(); sb.append(e == this ? "(this Collection)" : e); if (! i.hasNext()) return sb.append(']').toString(); sb.append(", "); } } @Override public boolean add(E e) { throw new UnsupportedOperationException("Not supported."); } @Override public void add(int index, E element) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean addAll(Collection<? extends E> c) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean addAll(int index, Collection<? extends E> c) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean addAll(E... elements) { throw new UnsupportedOperationException("Not supported."); } @Override public E set(int index, E element) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean setAll(Collection<? extends E> col) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean setAll(E... elements) { throw new UnsupportedOperationException("Not supported."); } @Override public void clear() { throw new UnsupportedOperationException("Not supported."); } @Override public E remove(int index) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean remove(Object o) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean removeAll(Collection<?> c) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean retainAll(Collection<?> c) { throw new UnsupportedOperationException("Not supported."); } @Override public void remove(int from, int to) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean removeAll(E... elements) { throw new UnsupportedOperationException("Not supported."); } @Override public boolean retainAll(E... elements) { throw new UnsupportedOperationException("Not supported."); } // Iterator to traverse the list of selected indices in both directions. private static class SelectionListIterator<E> implements ListIterator<E> { private int pos; private final ReadOnlyUnbackedObservableList<E> list; public SelectionListIterator(ReadOnlyUnbackedObservableList<E> list) { this(list, 0); } public SelectionListIterator(ReadOnlyUnbackedObservableList<E> list, int pos) { this.list = list; this.pos = pos; } @Override public boolean hasNext() { return pos < list.size(); } @Override public E next() { if (!hasNext()) { throw new NoSuchElementException(); } return list.get(pos++); } @Override public boolean hasPrevious() { return pos > 0; } @Override public E previous() { if (!hasPrevious()) { throw new NoSuchElementException(); } return list.get(--pos); } @Override public int nextIndex() { return pos; } @Override public int previousIndex() { return pos - 1; } @Override public void remove() { throw new UnsupportedOperationException("Not supported."); } @Override public void set(E e) { throw new UnsupportedOperationException("Not supported."); } @Override public void add(E e) { throw new UnsupportedOperationException("Not supported."); } } }