comparison org.eclipse.swt.win32.win32.x86/src/org/eclipse/swt/internal/image/OS2BMPFileFormat.d @ 0:6dd524f61e62

add dwt win and basic java stuff
author Frank Benoit <benoit@tionex.de>
date Mon, 02 Mar 2009 14:44:16 +0100
parents
children f36c67707cb3
comparison
equal deleted inserted replaced
-1:000000000000 0:6dd524f61e62
1 /*******************************************************************************
2 * Copyright (c) 2000, 2008 IBM Corporation and others.
3 * All rights reserved. This program and the accompanying materials
4 * are made available under the terms of the Eclipse Public License v1.0
5 * which accompanies this distribution, and is available at
6 * http://www.eclipse.org/legal/epl-v10.html
7 *
8 * Contributors:
9 * IBM Corporation - initial API and implementation
10 * Port to the D programming language:
11 * Frank Benoit <benoit@tionex.de>
12 *******************************************************************************/
13 module org.eclipse.swt.internal.image.OS2BMPFileFormat;
14
15 import org.eclipse.swt.SWT;
16 import org.eclipse.swt.graphics.ImageData;
17 import org.eclipse.swt.graphics.ImageLoader;
18 import org.eclipse.swt.graphics.PaletteData;
19 import org.eclipse.swt.graphics.RGB;
20 import org.eclipse.swt.internal.image.LEDataInputStream;
21 import org.eclipse.swt.internal.image.FileFormat;
22 import java.io.ByteArrayOutputStream;
23 import java.lang.all;
24
25 import tango.core.Exception;
26
27 final class OS2BMPFileFormat : FileFormat {
28 static final int BMPFileHeaderSize = 14;
29 static final int BMPHeaderFixedSize = 12;
30 int width, height, bitCount;
31
32 override bool isFileFormat(LEDataInputStream stream) {
33 try {
34 byte[] header = new byte[18];
35 stream.read(header);
36 stream.unread(header);
37 int infoHeaderSize = (header[14] & 0xFF) | ((header[15] & 0xFF) << 8) | ((header[16] & 0xFF) << 16) | ((header[17] & 0xFF) << 24);
38 return header[0] is 0x42 && header[1] is 0x4D && infoHeaderSize is BMPHeaderFixedSize;
39 } catch (Exception e) {
40 return false;
41 }
42 }
43 byte[] loadData(byte[] infoHeader) {
44 int stride = (width * bitCount + 7) / 8;
45 stride = (stride + 3) / 4 * 4; // Round up to 4 byte multiple
46 byte[] data = loadData(infoHeader, stride);
47 flipScanLines(data, stride, height);
48 return data;
49 }
50 byte[] loadData(byte[] infoHeader, int stride) {
51 int dataSize = height * stride;
52 byte[] data = new byte[dataSize];
53 try {
54 if (inputStream.read(data) !is dataSize)
55 SWT.error(SWT.ERROR_INVALID_IMAGE);
56 } catch (IOException e) {
57 SWT.error(SWT.ERROR_IO, e);
58 }
59 return data;
60 }
61 int[] loadFileHeader() {
62 int[] header = new int[5];
63 try {
64 header[0] = inputStream.readShort();
65 header[1] = inputStream.readInt();
66 header[2] = inputStream.readShort();
67 header[3] = inputStream.readShort();
68 header[4] = inputStream.readInt();
69 } catch (IOException e) {
70 SWT.error(SWT.ERROR_IO, e);
71 }
72 if (header[0] !is 0x4D42)
73 SWT.error(SWT.ERROR_INVALID_IMAGE);
74 return header;
75 }
76 override ImageData[] loadFromByteStream() {
77 int[] fileHeader = loadFileHeader();
78 byte[] infoHeader = new byte[BMPHeaderFixedSize];
79 try {
80 inputStream.read(infoHeader);
81 } catch (Exception e) {
82 SWT.error(SWT.ERROR_IO, e);
83 }
84 width = (infoHeader[4] & 0xFF) | ((infoHeader[5] & 0xFF) << 8);
85 height = (infoHeader[6] & 0xFF) | ((infoHeader[7] & 0xFF) << 8);
86 bitCount = (infoHeader[10] & 0xFF) | ((infoHeader[11] & 0xFF) << 8);
87 PaletteData palette = loadPalette(infoHeader);
88 if (inputStream.getPosition() < fileHeader[4]) {
89 // Seek to the specified offset
90 try {
91 inputStream.skip(fileHeader[4] - inputStream.getPosition());
92 } catch (IOException e) {
93 SWT.error(SWT.ERROR_IO, e);
94 }
95 }
96 byte[] data = loadData(infoHeader);
97 int type = SWT.IMAGE_OS2_BMP;
98 return [
99 ImageData.internal_new(
100 width,
101 height,
102 bitCount,
103 palette,
104 4,
105 data,
106 0,
107 null,
108 null,
109 -1,
110 -1,
111 type,
112 0,
113 0,
114 0,
115 0)
116 ];
117 }
118 PaletteData loadPalette(byte[] infoHeader) {
119 if (bitCount <= 8) {
120 int numColors = 1 << bitCount;
121 byte[] buf = new byte[numColors * 3];
122 try {
123 if (inputStream.read(buf) !is buf.length)
124 SWT.error(SWT.ERROR_INVALID_IMAGE);
125 } catch (IOException e) {
126 SWT.error(SWT.ERROR_IO, e);
127 }
128 return paletteFromBytes(buf, numColors);
129 }
130 if (bitCount is 16) return new PaletteData(0x7C00, 0x3E0, 0x1F);
131 if (bitCount is 24) return new PaletteData(0xFF, 0xFF00, 0xFF0000);
132 return new PaletteData(0xFF00, 0xFF0000, 0xFF000000);
133 }
134 PaletteData paletteFromBytes(byte[] bytes, int numColors) {
135 int bytesOffset = 0;
136 RGB[] colors = new RGB[numColors];
137 for (int i = 0; i < numColors; i++) {
138 colors[i] = new RGB(bytes[bytesOffset + 2] & 0xFF,
139 bytes[bytesOffset + 1] & 0xFF,
140 bytes[bytesOffset] & 0xFF);
141 bytesOffset += 3;
142 }
143 return new PaletteData(colors);
144 }
145 /**
146 * Answer a byte array containing the BMP representation of
147 * the given device independent palette.
148 */
149 static byte[] paletteToBytes(PaletteData pal) {
150 int n = pal.colors is null ? 0 : (pal.colors.length < 256 ? pal.colors.length : 256);
151 byte[] bytes = new byte[n * 3];
152 int offset = 0;
153 for (int i = 0; i < n; i++) {
154 RGB col = pal.colors[i];
155 bytes[offset] = cast(byte)col.blue;
156 bytes[offset + 1] = cast(byte)col.green;
157 bytes[offset + 2] = cast(byte)col.red;
158 offset += 3;
159 }
160 return bytes;
161 }
162 /**
163 * Unload the given image's data into the given byte stream.
164 * Answer the number of bytes written.
165 */
166 int unloadData(ImageData image, OutputStream ostr) {
167 int bmpBpl = 0;
168 try {
169 int bpl = (image.width * image.depth + 7) / 8;
170 bmpBpl = (bpl + 3) / 4 * 4; // BMP pads scanlines to multiples of 4 bytes
171 int linesPerBuf = 32678 / bmpBpl;
172 byte[] buf = new byte[linesPerBuf * bmpBpl];
173 byte[] data = image.data;
174 int imageBpl = image.bytesPerLine;
175 int dataIndex = imageBpl * (image.height - 1); // Start at last line
176 if (image.depth is 16) {
177 for (int y = 0; y < image.height; y += linesPerBuf) {
178 int count = image.height - y;
179 if (linesPerBuf < count) count = linesPerBuf;
180 int bufOffset = 0;
181 for (int i = 0; i < count; i++) {
182 for (int wIndex = 0; wIndex < bpl; wIndex += 2) {
183 buf[bufOffset + wIndex + 1] = data[dataIndex + wIndex + 1];
184 buf[bufOffset + wIndex] = data[dataIndex + wIndex];
185 }
186 bufOffset += bmpBpl;
187 dataIndex -= imageBpl;
188 }
189 ostr.write(buf, 0, bufOffset);
190 }
191 } else {
192 for (int y = 0; y < image.height; y += linesPerBuf) {
193 int tmp = image.height - y;
194 int count = tmp < linesPerBuf ? tmp : linesPerBuf;
195 int bufOffset = 0;
196 for (int i = 0; i < count; i++) {
197 System.arraycopy(data, dataIndex, buf, bufOffset, bpl);
198 bufOffset += bmpBpl;
199 dataIndex -= imageBpl;
200 }
201 ostr.write(buf, 0, bufOffset);
202 }
203 }
204 } catch (IOException e) {
205 SWT.error(SWT.ERROR_IO, e);
206 }
207 return bmpBpl * image.height;
208 }
209 /**
210 * Unload a DeviceIndependentImage using Windows .BMP format into the given
211 * byte stream.
212 */
213 override void unloadIntoByteStream(ImageLoader loader) {
214 ImageData image = loader.data[0];
215 byte[] rgbs;
216 int numCols;
217 if (!((image.depth is 1) || (image.depth is 4) || (image.depth is 8) ||
218 (image.depth is 16) || (image.depth is 24) || (image.depth is 32)))
219 SWT.error(SWT.ERROR_UNSUPPORTED_DEPTH);
220 PaletteData pal = image.palette;
221 if ((image.depth is 16) || (image.depth is 24) || (image.depth is 32)) {
222 if (!pal.isDirect)
223 SWT.error(SWT.ERROR_INVALID_IMAGE);
224 numCols = 0;
225 rgbs = null;
226 } else {
227 if (pal.isDirect)
228 SWT.error(SWT.ERROR_INVALID_IMAGE);
229 numCols = pal.colors.length;
230 rgbs = paletteToBytes(pal);
231 }
232 // Fill in file header, except for bfsize, which is done later.
233 int headersSize = BMPFileHeaderSize + BMPHeaderFixedSize;
234 int[] fileHeader = new int[5];
235 fileHeader[0] = 0x4D42; // Signature
236 fileHeader[1] = 0; // File size - filled in later
237 fileHeader[2] = 0; // Reserved 1
238 fileHeader[3] = 0; // Reserved 2
239 fileHeader[4] = headersSize; // Offset to data
240 if (rgbs !is null) {
241 fileHeader[4] += rgbs.length;
242 }
243
244 // Prepare data. This is done first so we don't have to try to rewind
245 // the stream and fill in the details later.
246 ByteArrayOutputStream ostr = new ByteArrayOutputStream();
247 unloadData(image, ostr);
248 byte[] data = ostr.toByteArray();
249
250 // Calculate file size
251 fileHeader[1] = fileHeader[4] + data.length;
252
253 // Write the headers
254 try {
255 outputStream.writeShort(fileHeader[0]);
256 outputStream.writeInt(fileHeader[1]);
257 outputStream.writeShort(fileHeader[2]);
258 outputStream.writeShort(fileHeader[3]);
259 outputStream.writeInt(fileHeader[4]);
260 } catch (IOException e) {
261 SWT.error(SWT.ERROR_IO, e);
262 }
263 try {
264 outputStream.writeInt(BMPHeaderFixedSize);
265 outputStream.writeShort(image.width);
266 outputStream.writeShort(image.height);
267 outputStream.writeShort(1);
268 outputStream.writeShort(cast(short)image.depth);
269 } catch (IOException e) {
270 SWT.error(SWT.ERROR_IO, e);
271 }
272
273 // Unload palette
274 if (numCols > 0) {
275 try {
276 outputStream.write(rgbs);
277 } catch (IOException e) {
278 SWT.error(SWT.ERROR_IO, e);
279 }
280 }
281
282 // Unload the data
283 try {
284 outputStream.write(data);
285 } catch (IOException e) {
286 SWT.error(SWT.ERROR_IO, e);
287 }
288 }
289 void flipScanLines(byte[] data, int stride, int height) {
290 int i1 = 0;
291 int i2 = (height - 1) * stride;
292 for (int i = 0; i < height / 2; i++) {
293 for (int index = 0; index < stride; index++) {
294 byte b = data[index + i1];
295 data[index + i1] = data[index + i2];
296 data[index + i2] = b;
297 }
298 i1 += stride;
299 i2 -= stride;
300 }
301 }
302
303 }