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

org.sejda.sambox.cos.COSArrayList Maven / Gradle / Ivy

Go to download

An Apache PDFBox fork intended to be used as PDF processor for Sejda and PDFsam related projects

There is a newer version: 3.0.24
Show newest version
/*
 * 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.sejda.sambox.cos;

import static java.util.Objects.nonNull;
import static java.util.Optional.ofNullable;

import java.util.*;

/**
 * This is an implementation of a List that will sync its contents to a COSArray.
 *
 * @author Ben Litchfield
 */
public class COSArrayList implements List
{
    private final COSArray array;
    private final List actual;

    // indicates that the list has been filtered
    // i.e. the number of entries in array and actual differ
    private boolean isFiltered = false;

    private COSDictionary parentDict;
    private COSName dictKey;

    /**
     * Default constructor.
     */
    public COSArrayList()
    {
        array = new COSArray();
        actual = new ArrayList<>();
    }

    /**
     * Constructor.
     *
     * @param actualList The list of standard java objects
     * @param cosArray The COS array object to sync to.
     */
    public COSArrayList(List actualList, COSArray cosArray)
    {
        actual = actualList;
        array = cosArray;

        // if the number of entries differs this may come from a filter being
        // applied at the PDModel level
        if (actual.size() != array.size()) {
            isFiltered = true;
        }
    }

    /**
     * This is a really special constructor. Sometimes the PDF spec says that a dictionary entry can either be a single
     * item or an array of those items. But in the PDModel interface we really just want to always return a
     * java.util.List. In the case were we get the list and never modify it we don't want to convert to COSArray and put
     * one element, unless we append to the list. So here we are going to create this object with a single item instead
     * of a list, but allow more items to be added and then converted to an array.
     *
     * @param actualObject The PDModel object.
     * @param item The COS Model object.
     * @param dictionary The dictionary that holds the item, and will hold the array if an item is added.
     * @param dictionaryKey The key into the dictionary to set the item.
     */
    public COSArrayList(E actualObject, COSBase item, COSDictionary dictionary,
            COSName dictionaryKey)
    {
        array = new COSArray();
        array.add(item);
        actual = new ArrayList<>();
        actual.add(actualObject);

        parentDict = dictionary;
        dictKey = dictionaryKey;
    }

    /**
     * This constructor is to be used if the array doesn't exist, but is to be created and added to the parent
     * dictionary as soon as the first element is added to the array.
     *
     * @param dictionary The dictionary that holds the item, and will hold the array if an item is added.
     * @param dictionaryKey The key into the dictionary to set the item.
     */
    public COSArrayList(COSDictionary dictionary, COSName dictionaryKey)
    {
        array = new COSArray();
        actual = new ArrayList<>();
        parentDict = dictionary;
        dictKey = dictionaryKey;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public int size()
    {
        return actual.size();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean isEmpty()
    {
        return actual.isEmpty();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean contains(Object o)
    {
        return actual.contains(o);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Iterator iterator()
    {
        return actual.iterator();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public Object[] toArray()
    {
        return actual.toArray();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public  X[] toArray(X[] a)
    {
        return actual.toArray(a);

    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean add(E o)
    {
        // when adding if there is a parentDict then change the item
        // in the dictionary from a single item to an array.
        if (parentDict != null)
        {
            parentDict.setItem(dictKey, array);
            // clear the parent dict so it doesn't happen again, there might be
            // a usecase for keeping the parentDict around but not now.
            parentDict = null;
        }
        // string is a special case because we can't subclass to be COSObjectable
        if (o instanceof String)
        {
            array.add(COSString.parseLiteral((String) o));
        }
        else
        {
            if (array != null)
            {
                array.add(((COSObjectable) o).getCOSObject());
            }
        }
        return actual.add(o);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean remove(Object o)
    {
        if (isFiltered) {
            throw new UnsupportedOperationException("removing entries from a filtered List is not permitted");
        }

        // the PDFBox approach based on toCOSObjectList(c) is flawed when migrated verbatim to Sambox,
        // due to how our codebase diverged
        // an index-based approach to removing works better

        boolean retval = true;
        int index = actual.indexOf(o);
        if(index >= 0)
        {
            actual.remove(index);
            array.remove(index);
        }
        else
        {
            retval = false;
        }
        return retval;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean containsAll(Collection c)
    {
        return actual.containsAll(c);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean addAll(Collection c)
    {
        // the PDFBox approach based on toCOSObjectList(c) is flawed when migrated verbatim to Sambox,
        // due to how our codebase diverged

        // add this back if needed/used
        throw new RuntimeException("Unsupported");
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean addAll(int index, Collection c)
    {
        // the PDFBox approach based on toCOSObjectList(c) is flawed when migrated verbatim to Sambox,
        // due to how our codebase diverged

        // add this back if needed/used
        throw new RuntimeException("Unsupported operation");
    }

    /**
     * This will take an array of COSNumbers and return a COSArrayList of java.lang.Integer values.
     *
     * @param intArray The existing integer Array.
     *
     * @return A list that is part of the core Java collections.
     */
    public static List convertIntegerCOSArrayToList(COSArray intArray)
    {
        List retval = null;
        if (intArray != null)
        {
            List numbers = new ArrayList<>();
            for (int i = 0; i < intArray.size(); i++)
            {
                COSNumber num = (COSNumber) intArray.get(i).getCOSObject();
                numbers.add(num.intValue());
            }
            retval = new COSArrayList<>(numbers, intArray);
        }
        return retval;
    }

    /**
     * This will take an array of COSNumbers and return a COSArrayList of java.lang.Float values.
     *
     * @param floatArray The existing float Array.
     *
     * @return The list of Float objects.
     */
    public static List convertFloatCOSArrayToList(COSArray floatArray)
    {
        if (nonNull(floatArray))
        {
            List numbers = new ArrayList<>(floatArray.size());
            for (int i = 0; i < floatArray.size(); i++)
            {
                numbers.add(ofNullable(floatArray.getObject(i)).filter(v -> v instanceof COSNumber)
                        .map(v -> ((COSNumber) v).floatValue()).orElse(null));
            }
            return new COSArrayList<>(numbers, floatArray);
        }
        return null;
    }

    /**
     * This will take an array of COSName and return a COSArrayList of java.lang.String values.
     *
     * @param nameArray The existing name Array.
     *
     * @return The list of String objects.
     */
    public static List convertCOSNameCOSArrayToList(COSArray nameArray)
    {
        List retval = null;
        if (nameArray != null)
        {
            List names = new ArrayList<>();
            for (int i = 0; i < nameArray.size(); i++)
            {
                names.add(((COSName) nameArray.getObject(i)).getName());
            }
            retval = new COSArrayList<>(names, nameArray);
        }
        return retval;
    }

    /**
     * This will take an array of COSString and return a COSArrayList of java.lang.String values.
     *
     * @param stringArray The existing name Array.
     *
     * @return The list of String objects.
     */
    public static List convertCOSStringCOSArrayToList(COSArray stringArray)
    {
        List retval = null;
        if (stringArray != null)
        {
            List string = new ArrayList<>();
            for (int i = 0; i < stringArray.size(); i++)
            {
                string.add(((COSString) stringArray.getObject(i)).getString());
            }
            retval = new COSArrayList<>(string, stringArray);
        }
        return retval;
    }

    /**
     * This will take an list of string objects and return a COSArray of COSName objects.
     *
     * @param strings A list of strings
     *
     * @return An array of COSName objects
     */
    public static COSArray convertStringListToCOSNameCOSArray(List strings)
    {
        COSArray retval = new COSArray();
        for (String string : strings)
        {
            retval.add(COSName.getPDFName(string));
        }
        return retval;
    }

    /**
     * This will take an list of string objects and return a COSArray of COSName objects.
     *
     * @param strings A list of strings
     *
     * @return An array of COSName objects
     */
    public static COSArray convertStringListToCOSStringCOSArray(List strings)
    {
        COSArray retval = new COSArray();
        for (String string : strings)
        {
            retval.add(COSString.parseLiteral(string));
        }
        return retval;
    }

    /**
     * This will convert a list of COSObjectables to an array list of COSBase objects.
     *
     * @param cosObjectableList A list of COSObjectable.
     *
     * @return An array of COSBase or null if the input list is null or empty.
     */
    public static COSArray converterToCOSArray(List cosObjectableList)
    {
        if (cosObjectableList != null && !cosObjectableList.isEmpty())
        {
            if (cosObjectableList instanceof COSArrayList)
            {
                // if it is already a COSArrayList then we don't want to recreate the array, we want to reuse it.
                return ((COSArrayList) cosObjectableList).array;
            }
            COSArray array = new COSArray();
            for (Object current : cosObjectableList)
            {
                if (current instanceof String)
                {
                    array.add(COSString.parseLiteral((String) current));
                }
                else if (current instanceof Integer || current instanceof Long)
                {
                    array.add(COSInteger.get(((Number) current).longValue()));
                }
                else if (current instanceof Float || current instanceof Double)
                {
                    array.add(new COSFloat(((Number) current).floatValue()));
                }
                else if (current instanceof COSObjectable)
                {
                    array.add(((COSObjectable) current).getCOSObject());
                }
                else if (current == null)
                {
                    array.add(COSNull.NULL);
                }
                else
                {
                    throw new RuntimeException(
                            "Unable to convert '" + current.getClass().getName() + "' to COSBase ");
                }
            }
            return array;
        }
        return null;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean removeAll(Collection c)
    {
        // the PDFBox approach based on toCOSObjectList(c) is flawed when migrated verbatim to Sambox,
        // due to how our codebase diverged:

        // in the case of Annotations, page.getAnnotations(), the array is filled with ExistingCOSIndirectObject entries
        // while actual contains PDAnnotation entries
        // toCOSObjectList(PDAnnotation) does not return a ExistingCOSIndirectObject so
        // they are out of sync and all remove/removeAll operations silently and subtly fail by
        // causing the 2 arrays to go out of sync

        // annotations = page.getAnnotations();
        // annotations.size == 3

        // annotations.remove(annotations.get(0)) // only removes from actual, not array
        // annotations.array.size() == 3
        // annotations.actual.size == 2

        // page.setAnnotations(annotations)
        // page.getAnnotations().size() == 3 (! expected 2)

        // another issue is page.getAnnotations(filter) accepts a filter that can
        // make actual and array be out of sync from the start
        // operations that mutate the COSArrayList should not be allowed in this case

        //array.removeAll(toCOSObjectList(c));
        //return actual.removeAll(c);

        if (isFiltered)
        {
            throw new UnsupportedOperationException("removing entries from a filtered List is not permitted");
        }

        boolean retval = false;
        for(Object o: c)
        {
            retval =  retval || remove(o);
        }
        return retval;
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean retainAll(Collection c)
    {
        if (isFiltered)
        {
            throw new UnsupportedOperationException("removing entries from a filtered List is not permitted");
        }

        List toRemove = new LinkedList<>();
        for(E o: actual)
        {
            if (!c.contains(o))
            {
                // remove element
                toRemove.add(o);
            }
        }

        return removeAll(toRemove);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public void clear()
    {
        // when adding if there is a parentDict then change the item
        // in the dictionary from a single item to an array.
        if (parentDict != null)
        {
            parentDict.setItem(dictKey, (COSBase) null);
        }
        actual.clear();
        array.clear();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean equals(Object o)
    {
        return actual.equals(o);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public int hashCode()
    {
        return actual.hashCode();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public E get(int index)
    {
        return actual.get(index);

    }

    /**
     * {@inheritDoc}
     */
    @Override
    public E set(int index, E element)
    {
        if (element instanceof String)
        {
            COSString item = COSString.parseLiteral((String) element);
            if (parentDict != null && index == 0)
            {
                parentDict.setItem(dictKey, item);
            }
            array.set(index, item);
        }
        else
        {
            if (parentDict != null && index == 0)
            {
                parentDict.setItem(dictKey, ((COSObjectable) element).getCOSObject());
            }
            array.set(index, ((COSObjectable) element).getCOSObject());
        }
        return actual.set(index, element);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public void add(int index, E element)
    {
        // when adding if there is a parentDict then change the item
        // in the dictionary from a single item to an array.
        if (parentDict != null)
        {
            parentDict.setItem(dictKey, array);
            // clear the parent dict so it doesn't happen again, there might be
            // a usecase for keeping the parentDict around but not now.
            parentDict = null;
        }
        actual.add(index, element);
        if (element instanceof String)
        {
            array.add(index, COSString.parseLiteral((String) element));
        }
        else
        {
            array.add(index, ((COSObjectable) element).getCOSObject());
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public E remove(int index)
    {
        if (isFiltered) {
            throw new UnsupportedOperationException("removing entries from a filtered List is not permitted");
        }

        array.remove(index);
        return actual.remove(index);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public int indexOf(Object o)
    {
        return actual.indexOf(o);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public int lastIndexOf(Object o)
    {
        return actual.lastIndexOf(o);

    }

    /**
     * {@inheritDoc}
     */
    @Override
    public ListIterator listIterator()
    {
        return actual.listIterator();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public ListIterator listIterator(int index)
    {
        return actual.listIterator(index);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public List subList(int fromIndex, int toIndex)
    {
        return actual.subList(fromIndex, toIndex);
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public String toString()
    {
        return "COSArrayList{" + array.toString() + "}";
    }

    /**
     * This will return the underlying COSArray.
     *
     * @return the COSArray
     */
    public COSArray getCOSArray()
    {
        return array;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy