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

org.sejda.sambox.output.ContentStreamWriter 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.21
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.output;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.List;
import java.util.Optional;

import org.apache.fontbox.util.Charsets;
import org.sejda.io.BufferedCountingChannelWriter;
import org.sejda.io.CountingWritableByteChannel;
import org.sejda.sambox.contentstream.operator.Operator;
import org.sejda.sambox.contentstream.operator.OperatorName;
import org.sejda.sambox.cos.COSBase;
import org.sejda.sambox.cos.COSDictionary;
import org.sejda.sambox.cos.COSName;
import org.sejda.sambox.cos.COSStream;
import org.sejda.sambox.cos.IndirectCOSObjectReference;

/**
 * Component capable of writing a content stream tokens, {@link Operator}s and {@link COSBase} operands.
 * 
 * @author Andrea Vacondio
 */
public class ContentStreamWriter extends DefaultCOSWriter
{

    public ContentStreamWriter(CountingWritableByteChannel channel)
    {
        super(channel);
    }

    public ContentStreamWriter(BufferedCountingChannelWriter writer)
    {
        super(writer);
    }

    public void writeTokens(List tokens) throws IOException
    {
        for (Object token : tokens)
        {
            if (token instanceof COSBase)
            {
                ((COSBase) token).accept(this);
                writeSpace();
            }
            else if (token instanceof Operator)
            {
                this.writeOperator((Operator) token);
            }
            else
            {
                throw new IOException("Unsupported type in content stream:" + token);
            }
        }
    }

    public void writeTokens(Operator... tokens) throws IOException
    {
        for (Operator token : tokens)
        {
            writeOperator(token);
        }
    }

    public void writeOperator(List operands, Operator operator) throws IOException
    {
        for (COSBase operand : operands)
        {
            operand.accept(this);
            writeSpace();
        }
        this.writeOperator(operator);
    }

    /**
     * Writes the byte array as is as content of the stream.
     * 
     * @param byteArray
     * @throws IOException
     */
    public void writeContent(byte[] byteArray) throws IOException
    {
        writer().write(byteArray);
    }

    public void writeEOL() throws IOException
    {
        writer().writeEOL();
    }

    public void writeSpace() throws IOException
    {
        writer().write(SPACE);
    }

    public void writeComment(String comment) throws IOException
    {
        writer().write(PERCENT_SIGN);
        writeContent(comment.getBytes(Charsets.US_ASCII));
        writeEOL();
    }

    private void writeOperator(Operator token) throws IOException
    {
        writer().write(token.getName().getBytes(StandardCharsets.ISO_8859_1));
        if (token.getName().equals(OperatorName.BEGIN_INLINE_IMAGE))
        {
            writeEOL();
            COSDictionary imageParams = Optional.ofNullable(token.getImageParameters())
                    .orElseGet(COSDictionary::new);
            for (COSName key : imageParams.keySet())
            {
                key.accept(this);
                writeSpace();
                COSBase imageParamsDictionaryObject = imageParams.getDictionaryObject(key);
                if (imageParamsDictionaryObject != null)
                {
                    imageParamsDictionaryObject.accept(this);
                }
                writeEOL();
            }
            writer().write(
                    OperatorName.BEGIN_INLINE_IMAGE_DATA.getBytes(StandardCharsets.US_ASCII));
            writeEOL();
            writer().write(token.getImageData());
            writeEOL();
            writer().write(OperatorName.END_INLINE_IMAGE.getBytes(StandardCharsets.US_ASCII));
        }
        writeEOL();
    }

    @Override
    public void visit(COSStream value)
    {
        throw new UnsupportedOperationException("Cannot write a stream inside a stream");
    }

    @Override
    public void visit(IndirectCOSObjectReference value)
    {
        throw new UnsupportedOperationException(
                "Cannot write an indirect object reference inside a stream");
    }

    @Override
    public void writeComplexObjectSeparator()
    {
        // write nothing
    }
}