/* * 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 javax.imageio.stream; import java.io.IOException; import java.io.File; import java.io.OutputStream; import java.io.RandomAccessFile; /** * The FileCacheImageOutputStream class is an implementation of * ImageOutputStream that writes to its OutputStream using a temporary file as a * cache. * * @since Android 1.0 */ public class FileCacheImageOutputStream extends ImageOutputStreamImpl { /** * The Constant IIO_TEMP_FILE_PREFIX. */ static final String IIO_TEMP_FILE_PREFIX = "iioCache"; /** * The Constant MAX_BUFFER_LEN. */ static final int MAX_BUFFER_LEN = 1048575; // 1 MB - is it not too much? /** * The os. */ private OutputStream os; /** * The file. */ private File file; /** * The raf. */ private RandomAccessFile raf; /** * Instantiates a FileCacheImageOutputStream. * * @param stream * the OutputStream for writing. * @param cacheDir * the cache directory where the cache file will be created. * @throws IOException * if an I/O exception has occurred. */ public FileCacheImageOutputStream(OutputStream stream, File cacheDir) throws IOException { if (stream == null) { throw new IllegalArgumentException("stream == null!"); } os = stream; if (cacheDir == null || cacheDir.isDirectory()) { file = File.createTempFile(IIO_TEMP_FILE_PREFIX, null, cacheDir); file.deleteOnExit(); } else { throw new IllegalArgumentException("Not a directory!"); } raf = new RandomAccessFile(file, "rw"); } @Override public void close() throws IOException { flushBefore(raf.length()); super.close(); raf.close(); file.delete(); } @Override public boolean isCached() { return true; } @Override public boolean isCachedFile() { return true; } @Override public boolean isCachedMemory() { return false; } @Override public void write(int b) throws IOException { flushBits(); // See the flushBits method description raf.write(b); streamPos++; } @Override public void write(byte[] b, int off, int len) throws IOException { flushBits(); // See the flushBits method description raf.write(b, off, len); streamPos += len; } @Override public int read() throws IOException { bitOffset = 0; // Should reset int res = raf.read(); if (res >= 0) { streamPos++; } return res; } @Override public int read(byte[] b, int off, int len) throws IOException { bitOffset = 0; int numRead = raf.read(b, off, len); if (numRead > 0) { streamPos += numRead; } return numRead; } @Override public void flushBefore(long pos) throws IOException { long readFromPos = flushedPos; super.flushBefore(pos); long bytesToRead = pos - readFromPos; raf.seek(readFromPos); if (bytesToRead < MAX_BUFFER_LEN) { byte buffer[] = new byte[(int)bytesToRead]; raf.readFully(buffer); os.write(buffer); } else { byte buffer[] = new byte[MAX_BUFFER_LEN]; while (bytesToRead > 0) { int count = (int)Math.min(MAX_BUFFER_LEN, bytesToRead); raf.readFully(buffer, 0, count); os.write(buffer, 0, count); bytesToRead -= count; } } os.flush(); if (pos != streamPos) { raf.seek(streamPos); // Reset the position } } @Override public void seek(long pos) throws IOException { if (pos < flushedPos) { throw new IndexOutOfBoundsException(); } raf.seek(pos); streamPos = raf.getFilePointer(); bitOffset = 0; } @Override public long length() { try { return raf.length(); } catch (IOException e) { return -1L; } } }