dmolony-DiskBrowser/src/com/bytezone/diskbrowser/applefile/HiResImage.java

491 lines
11 KiB
Java
Raw Normal View History

2016-12-31 09:34:15 +00:00
package com.bytezone.diskbrowser.applefile;
2017-01-26 04:19:23 +00:00
import java.awt.Color;
2016-12-31 09:34:15 +00:00
import java.io.ByteArrayInputStream;
import java.io.IOException;
2017-01-14 01:40:58 +00:00
import java.util.List;
2016-12-31 09:34:15 +00:00
import javax.imageio.ImageIO;
import com.bytezone.diskbrowser.prodos.ProdosConstants;
2017-01-26 04:19:23 +00:00
import com.bytezone.diskbrowser.utilities.HexFormatter;
2016-12-31 09:34:15 +00:00
public abstract class HiResImage extends AbstractFile
{
2017-01-14 01:40:58 +00:00
protected static PaletteFactory paletteFactory = new PaletteFactory ();
2017-01-12 22:11:05 +00:00
2016-12-31 09:34:15 +00:00
private static final byte[] pngHeader =
{ (byte) 0x89, 0x50, 0x4E, 0x47, 0x0D, 0x0A, 0x1A, 0x0A };
protected static boolean colourQuirks;
protected static boolean monochrome;
protected int fileType;
protected int auxType;
2017-01-25 11:02:50 +00:00
protected int eof;
2016-12-31 09:34:15 +00:00
protected byte[] unpackedBuffer;
2017-01-14 20:58:07 +00:00
protected int paletteIndex;
2016-12-31 09:34:15 +00:00
public HiResImage (String name, byte[] buffer)
{
super (name, buffer);
}
public HiResImage (String name, byte[] buffer, int loadAddress)
{
this (name, buffer, loadAddress, false);
}
public HiResImage (String name, byte[] buffer, int loadAddress, boolean scrunched)
{
super (name, buffer);
this.loadAddress = loadAddress; // for the disassembly listing
if (scrunched)
this.buffer = unscrunch (buffer);
}
2017-01-25 11:02:50 +00:00
public HiResImage (String name, byte[] buffer, int fileType, int auxType, int eof)
2016-12-31 09:34:15 +00:00
{
super (name, buffer);
this.fileType = fileType;
this.auxType = auxType;
2017-01-25 11:02:50 +00:00
this.eof = eof;
2016-12-31 09:34:15 +00:00
}
protected void createImage ()
{
2017-01-25 05:26:37 +00:00
if (isGif (buffer) || isPng (buffer) || isBmp (buffer))
2016-12-31 09:34:15 +00:00
makeImage ();
else if (monochrome)
createMonochromeImage ();
else
createColourImage ();
}
protected abstract void createMonochromeImage ();
protected abstract void createColourImage ();
2017-01-14 20:58:07 +00:00
public void checkPalette ()
{
if (!monochrome && paletteIndex != paletteFactory.getCurrentPaletteIndex ())
createImage ();
}
2017-01-14 11:11:07 +00:00
public void setPalette ()
2017-01-11 21:26:28 +00:00
{
if (!monochrome)
createImage ();
}
2016-12-31 09:34:15 +00:00
public void setColourQuirks (boolean value)
{
if (colourQuirks == value)
return;
colourQuirks = value;
if (!monochrome)
createImage ();
}
public void setMonochrome (boolean value)
{
if (monochrome == value)
return;
monochrome = value;
createImage ();
}
public static void setDefaultColourQuirks (boolean value)
{
colourQuirks = value;
}
public static void setDefaultMonochrome (boolean value)
{
monochrome = value;
}
/*-
Mode Page 1 Page 2
280 x 192 Black & White 0 4
280 x 192 Limited Color 1 5
560 x 192 Black & White 2 6
140 x 192 Full Color 3 7
*/
// SHR see - http://noboot.com/charlie/cb2e_p3.htm
@Override
public String getText ()
{
String auxText = "";
StringBuilder text = new StringBuilder ("Image File : " + name);
2017-01-26 04:19:23 +00:00
text.append (String.format ("%nFile type : $%02X %s", fileType,
ProdosConstants.fileTypes[fileType]));
2016-12-31 09:34:15 +00:00
switch (fileType)
{
2017-01-26 04:19:23 +00:00
case ProdosConstants.FILE_TYPE_PICT: // 0x08
2016-12-31 09:34:15 +00:00
if (auxType < 0x4000)
{
2017-01-26 04:19:23 +00:00
auxText = "Apple II Graphics File";
2016-12-31 09:34:15 +00:00
byte mode = buffer[0x78]; // 0-7
2017-01-26 04:19:23 +00:00
System.out.println ("Prodos PICT, mode=" + mode); // see mode table above
2016-12-31 09:34:15 +00:00
}
else if (auxType == 0x4000)
auxText = "Packed Hi-Res File";
else if (auxType == 0x4001)
auxText = "Packed Double Hi-Res File";
2017-01-27 07:11:00 +00:00
else
auxText = "Unknown aux: " + auxType;
2016-12-31 09:34:15 +00:00
break;
2017-01-26 04:19:23 +00:00
case ProdosConstants.FILE_TYPE_PNT: // 0xC0
if (auxType == 0)
auxText = "Paintworks Packed SHR Image";
else if (auxType == 1)
2016-12-31 09:34:15 +00:00
auxText = "Packed Super Hi-Res Image";
else if (auxType == 2)
2017-01-25 01:38:00 +00:00
auxText = "Super Hi-Res Image (Apple Preferred)";
2016-12-31 09:34:15 +00:00
else if (auxType == 3)
auxText = "Packed QuickDraw II PICT File";
2017-01-27 07:11:00 +00:00
else
auxText = "Unknown aux: " + auxType;
2016-12-31 09:34:15 +00:00
break;
2017-01-26 04:19:23 +00:00
case ProdosConstants.FILE_TYPE_PIC: // 0xC1
2016-12-31 09:34:15 +00:00
if (auxType == 0)
auxText = "Super Hi-res Screen Image";
else if (auxType == 1)
auxText = "QuickDraw PICT File";
else if (auxType == 2)
auxText = "Super Hi-Res 3200 color image";
2017-01-27 07:11:00 +00:00
else
auxText = "Unknown aux: " + auxType;
2016-12-31 09:34:15 +00:00
}
if (!auxText.isEmpty ())
text.append (String.format ("%nAux type : $%04X %s", auxType, auxText));
text.append (String.format ("%nFile size : %,d", buffer.length));
2017-01-25 11:02:50 +00:00
text.append (String.format ("%nEOF : %,d", eof));
2016-12-31 09:34:15 +00:00
if (unpackedBuffer != null)
{
text.append (String.format ("%nUnpacked : %,d%n%n", unpackedBuffer.length));
2017-01-25 01:38:00 +00:00
// text.append (HexFormatter.format (unpackedBuffer));
2016-12-31 09:34:15 +00:00
}
return text.toString ();
}
2017-01-25 04:40:15 +00:00
/*
* Unpack the Apple PackBytes format.
*
* Format is:
* <flag><data> ...
*
* Flag values (first 6 bits of flag byte):
* 00xxxxxx: (0-63) 1 to 64 bytes follow, all different
* 01xxxxxx: (0-63) 1 to 64 repeats of next byte
* 10xxxxxx: (0-63) 1 to 64 repeats of next 4 bytes
* 11xxxxxx: (0-63) 1 to 64 repeats of next byte taken as 4 bytes
* (as in 10xxxxxx case)
*/
2016-12-31 09:34:15 +00:00
// Super Hi-res IIGS
protected byte[] unpackBytes (byte[] buffer)
{
// routine found here - http://kpreid.livejournal.com/4319.html
2017-01-26 04:19:23 +00:00
byte[] newBuf = new byte[32768]; // this might be wrong
2016-12-31 09:34:15 +00:00
byte[] fourBuf = new byte[4];
int ptr = 0, newPtr = 0;
while (ptr < buffer.length)
{
2017-01-24 08:59:40 +00:00
int type = (buffer[ptr] & 0xC0) >> 6; // 0-3
int count = (buffer[ptr++] & 0x3F) + 1; // 1-64
2016-12-31 09:34:15 +00:00
switch (type)
{
case 0:
while (count-- != 0)
newBuf[newPtr++] = buffer[ptr++];
break;
case 1:
byte b = buffer[ptr++];
while (count-- != 0)
newBuf[newPtr++] = b;
break;
case 2:
for (int i = 0; i < 4; i++)
fourBuf[i] = buffer[ptr++];
while (count-- != 0)
for (int i = 0; i < 4; i++)
newBuf[newPtr++] = fourBuf[i];
break;
case 3:
b = buffer[ptr++];
count *= 4;
while (count-- != 0)
newBuf[newPtr++] = b;
break;
}
}
return newBuf;
}
2017-01-26 04:19:23 +00:00
// Super Hi-res IIGS
protected int unpackLine (byte[] buffer, byte[] newBuf, int newPtr)
{
byte[] fourBuf = new byte[4];
int ptr = 0;
while (ptr < buffer.length)
{
int type = (buffer[ptr] & 0xC0) >> 6; // 0-3
int count = (buffer[ptr++] & 0x3F) + 1; // 1-64
if (ptr >= buffer.length)
break;
switch (type)
{
case 0:
while (count-- != 0)
if (newPtr < unpackedBuffer.length && ptr < buffer.length)
newBuf[newPtr++] = buffer[ptr++];
break;
case 1:
byte b = buffer[ptr++];
while (count-- != 0)
if (newPtr < unpackedBuffer.length)
newBuf[newPtr++] = b;
break;
case 2:
for (int i = 0; i < 4; i++)
if (ptr < buffer.length)
fourBuf[i] = buffer[ptr++];
while (count-- != 0)
for (int i = 0; i < 4; i++)
if (newPtr < unpackedBuffer.length)
newBuf[newPtr++] = fourBuf[i];
break;
case 3:
b = buffer[ptr++];
count *= 4;
while (count-- != 0)
if (newPtr < unpackedBuffer.length)
newBuf[newPtr++] = b;
break;
}
}
return newPtr;
}
2016-12-31 09:34:15 +00:00
// Beagle Bros routine to expand a hi-res screen
private byte[] unscrunch (byte[] src)
{
byte[] dst = new byte[0x2000];
int p1 = 0;
int p2 = 0;
while (p1 < dst.length)
{
byte b = src[p2++];
if ((b == (byte) 0x80) || (b == (byte) 0xFF))
{
b &= 0x7F;
int rpt = src[p2++];
for (int i = 0; i < rpt; i++)
dst[p1++] = b;
}
else
dst[p1++] = b;
}
return dst;
}
protected void makeImage ()
{
try
{
image = ImageIO.read (new ByteArrayInputStream (buffer));
}
catch (IOException e)
{
e.printStackTrace ();
}
2017-01-25 05:26:37 +00:00
catch (IndexOutOfBoundsException e) // some BMP files cause this
{
System.out.println ("Error in makeImage()");
System.out.println (e.getMessage ());
}
2016-12-31 09:34:15 +00:00
}
public static boolean isGif (byte[] buffer)
{
if (buffer.length < 6)
return false;
String text = new String (buffer, 0, 6);
return text.equals ("GIF89a") || text.equals ("GIF87a");
}
public static boolean isPng (byte[] buffer)
{
if (buffer.length < pngHeader.length)
return false;
for (int i = 0; i < pngHeader.length; i++)
if (pngHeader[i] != buffer[i])
return false;
return true;
}
2017-01-12 22:11:05 +00:00
2017-01-25 05:26:37 +00:00
public static boolean isBmp (byte[] buffer)
{
if (buffer.length < 2)
return false;
String text = new String (buffer, 0, 2);
return text.equals ("BM");
}
2017-01-14 01:40:58 +00:00
public static PaletteFactory getPaletteFactory ()
2017-01-12 22:11:05 +00:00
{
2017-01-14 01:40:58 +00:00
return paletteFactory;
2017-01-12 22:11:05 +00:00
}
2017-01-14 01:40:58 +00:00
public static List<Palette> getPalettes ()
2017-01-12 22:11:05 +00:00
{
2017-01-14 01:40:58 +00:00
return paletteFactory.getPalettes ();
2017-01-12 22:11:05 +00:00
}
2017-01-26 04:19:23 +00:00
class ColorTable
{
int id;
ColorEntry[] entries = new ColorEntry[16];
public ColorTable ()
{
// default empty table
id = -1;
for (int i = 0; i < 16; i++)
{
entries[i] = new ColorEntry ();
}
}
public ColorTable (int id, byte[] data, int offset)
{
this.id = id;
for (int i = 0; i < 16; i++)
{
entries[i] = new ColorEntry (data, offset);
offset += 2;
}
}
String toLine ()
{
StringBuilder text = new StringBuilder ();
text.append (String.format (" %X", id));
for (int i = 0; i < 16; i++)
text.append (String.format (" %04X", entries[i].value));
return text.toString ();
}
void reverse ()
{
for (int i = 0; i < 8; i++)
{
ColorEntry temp = entries[i];
entries[i] = entries[15 - i];
entries[15 - i] = temp;
}
}
@Override
public String toString ()
{
StringBuilder text = new StringBuilder ();
text.append (String.format ("%2d ColorTable%n", id));
for (int i = 0; i < 8; i++)
text.append (String.format (" %2d: %04X", i, entries[i].value));
text.append ("\n");
for (int i = 8; i < 16; i++)
text.append (String.format (" %2d: %04X", i, entries[i].value));
return text.toString ();
}
}
class ColorEntry
{
int value; // 0RGB
Color color;
public ColorEntry ()
{
// default empty entry
value = 0;
color = new Color (0, 0, 0);
}
public ColorEntry (byte[] data, int offset)
{
value = HexFormatter.unsignedShort (data, offset);
int red = ((value >> 8) & 0x0f) * 17;
int green = ((value >> 4) & 0x0f) * 17;
int blue = (value & 0x0f) * 17;
2017-01-27 07:11:00 +00:00
2017-01-26 04:19:23 +00:00
color = new Color (red, green, blue);
}
@Override
public String toString ()
{
return String.format ("ColorEntry: %04X", value);
}
}
class DirEntry
{
int numBytes;
int mode;
public DirEntry (byte[] data, int offset)
{
numBytes = HexFormatter.unsignedShort (data, offset);
mode = HexFormatter.unsignedShort (data, offset + 2);
}
@Override
public String toString ()
{
return String.format ("Bytes: %5d, mode: %02X", numBytes, mode);
}
}
2015-06-01 09:35:51 +00:00
}