001/* 
002    Licensed to the Apache Software Foundation (ASF) under one
003    or more contributor license agreements.  See the NOTICE file
004    distributed with this work for additional information
005    regarding copyright ownership.  The ASF licenses this file
006    to you under the Apache License, Version 2.0 (the
007    "License"); you may not use this file except in compliance
008    with the License.  You may obtain a copy of the License at
009
010       http://www.apache.org/licenses/LICENSE-2.0
011
012    Unless required by applicable law or agreed to in writing,
013    software distributed under the License is distributed on an
014    "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
015    KIND, either express or implied.  See the License for the
016    specific language governing permissions and limitations
017    under the License.  
018 */
019package org.apache.wiki.util;
020
021import java.util.AbstractList;
022import java.util.concurrent.CopyOnWriteArrayList;
023
024/**
025 *  Builds a simple, priority-based List implementation.  The list
026 *  will be sorted according to the priority.  If two items are
027 *  inserted with the same priority, their order is the insertion order - i.e. the new one
028 *  is appended last in the insertion list.
029 *  <p>
030 *  Priority is an integer, and the list is sorted in descending order
031 *  (that is, 100 is before 10 is before 0 is before -40).
032 */
033public class PriorityList<E>
034    extends AbstractList<E>
035{
036    private final CopyOnWriteArrayList<Item<E>> m_elements = new CopyOnWriteArrayList<Item<E>>();
037
038    /**
039     *  This is the default priority, which is used if no priority
040     *  is defined.  It's current value is zero.
041     */
042    public static final int DEFAULT_PRIORITY = 0;
043
044    /**
045     *  Adds an object to its correct place in the list, using the
046     *  given priority.
047     *
048     *  @param o Object to add.
049     *  @param priority Priority.
050     */
051    public void add(final E o, final int priority )
052    {
053        int i = 0;
054
055        for( ; i < m_elements.size(); i++ )
056        {
057            final Item< E > item = m_elements.get(i);
058
059            if( item.m_priority < priority )
060            {
061                break;
062            }
063        }
064
065        final Item<E> newItem = new Item<E>();
066        newItem.m_priority = priority;
067        newItem.m_object   = o;
068
069        m_elements.add( i, newItem );
070    }
071
072    /**
073     *  Adds an object using the default priority to the List.
074     *
075     *  @param o Object to add.
076     *  @return true, as per the general Collections.add contract.
077     */
078    public boolean add(final E o )
079    {
080        add( o, DEFAULT_PRIORITY );
081
082        return true;
083    }
084
085    /**
086     *  Returns the object at index "index".
087     *
088     *  @param index The index.
089     *  @return The object at the list at the position "index".
090     */
091    public E get(final int index )
092    {
093        return m_elements.get( index ).m_object;
094    }
095
096    /**
097     *  Returns the current size of the list.
098     *  
099     *  @return size of the list.
100     */
101    public int size()
102    {
103        return m_elements.size();
104    }
105
106    /**
107     *  Provides a holder for the priority-object 2-tuple.
108     */
109    private static class Item<E>
110    {
111        public int     m_priority;
112        public E       m_object;
113    }
114}