blob: 96546662d818278f1ddc9acddaaf035731f9025a [file] [log] [blame]
/*
* Copyright (c) 1995, 2003, 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.
*/
/*-
* Reads xbitmap format images into a DIBitmap structure.
*/
package sun.awt.image;
import java.io.*;
import java.awt.image.*;
/**
* Parse files of the form:
*
* #define foo_width w
* #define foo_height h
* static char foo_bits[] = {
* 0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,
* 0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,0xnn,
* 0xnn,0xnn,0xnn,0xnn};
*
* @author James Gosling
*/
public class XbmImageDecoder extends ImageDecoder {
private static byte XbmColormap[] = {(byte) 255, (byte) 255, (byte) 255,
0, 0, 0};
private static int XbmHints = (ImageConsumer.TOPDOWNLEFTRIGHT |
ImageConsumer.COMPLETESCANLINES |
ImageConsumer.SINGLEPASS |
ImageConsumer.SINGLEFRAME);
public XbmImageDecoder(InputStreamImageSource src, InputStream is) {
super(src, is);
if (!(input instanceof BufferedInputStream)) {
// If the topmost stream is a metered stream,
// we take forever to decode the image...
input = new BufferedInputStream(input, 80);
}
}
/**
* An error has occurred. Throw an exception.
*/
private static void error(String s1) throws ImageFormatException {
throw new ImageFormatException(s1);
}
/**
* produce an image from the stream.
*/
public void produceImage() throws IOException, ImageFormatException {
char nm[] = new char[80];
int c;
int i = 0;
int state = 0;
int H = 0;
int W = 0;
int x = 0;
int y = 0;
boolean start = true;
byte raster[] = null;
IndexColorModel model = null;
while (!aborted && (c = input.read()) != -1) {
if ('a' <= c && c <= 'z' ||
'A' <= c && c <= 'Z' ||
'0' <= c && c <= '9' || c == '#' || c == '_') {
if (i < 78)
nm[i++] = (char) c;
} else if (i > 0) {
int nc = i;
i = 0;
if (start) {
if (nc != 7 ||
nm[0] != '#' ||
nm[1] != 'd' ||
nm[2] != 'e' ||
nm[3] != 'f' ||
nm[4] != 'i' ||
nm[5] != 'n' ||
nm[6] != 'e')
{
error("Not an XBM file");
}
start = false;
}
if (nm[nc - 1] == 'h')
state = 1; /* expecting width */
else if (nm[nc - 1] == 't' && nc > 1 && nm[nc - 2] == 'h')
state = 2; /* expecting height */
else if (nc > 2 && state < 0 && nm[0] == '0' && nm[1] == 'x') {
int n = 0;
for (int p = 2; p < nc; p++) {
c = nm[p];
if ('0' <= c && c <= '9')
c = c - '0';
else if ('A' <= c && c <= 'Z')
c = c - 'A' + 10;
else if ('a' <= c && c <= 'z')
c = c - 'a' + 10;
else
c = 0;
n = n * 16 + c;
}
for (int mask = 1; mask <= 0x80; mask <<= 1) {
if (x < W) {
if ((n & mask) != 0)
raster[x] = 1;
else
raster[x] = 0;
}
x++;
}
if (x >= W) {
if (setPixels(0, y, W, 1, model, raster, 0, W) <= 0) {
return;
}
x = 0;
if (y++ >= H) {
break;
}
}
} else {
int n = 0;
for (int p = 0; p < nc; p++)
if ('0' <= (c = nm[p]) && c <= '9')
n = n * 10 + c - '0';
else {
n = -1;
break;
}
if (n > 0 && state > 0) {
if (state == 1)
W = n;
else
H = n;
if (W == 0 || H == 0)
state = 0;
else {
model = new IndexColorModel(8, 2, XbmColormap,
0, false, 0);
setDimensions(W, H);
setColorModel(model);
setHints(XbmHints);
headerComplete();
raster = new byte[W];
state = -1;
}
}
}
}
}
input.close();
imageComplete(ImageConsumer.STATICIMAGEDONE, true);
}
}