enikey | 00e0dbf | 2008-12-08 10:59:59 +0000 | [diff] [blame] | 1 | package com.tightvnc.decoder; |
| 2 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 3 | import com.tightvnc.vncviewer.RecordInterface; |
| 4 | import com.tightvnc.vncviewer.RfbInputStream; |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 5 | import java.io.IOException; |
enikey | c6f6bab | 2008-12-24 03:41:00 +0000 | [diff] [blame] | 6 | import java.io.DataOutput; |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 7 | import java.awt.Graphics; |
| 8 | import java.awt.Image; |
| 9 | import java.awt.image.ColorModel; |
| 10 | import java.awt.image.DirectColorModel; |
| 11 | import java.awt.image.MemoryImageSource; |
| 12 | import java.awt.Color; |
enikey | 663025d | 2008-12-19 05:14:15 +0000 | [diff] [blame] | 13 | import java.awt.Toolkit; |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 14 | |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 15 | // |
| 16 | // This is base decoder class. |
| 17 | // Other classes will be childs of RawDecoder. |
| 18 | // |
enikey | ccf6e7c | 2008-12-19 06:00:18 +0000 | [diff] [blame] | 19 | |
enikey | 00e0dbf | 2008-12-08 10:59:59 +0000 | [diff] [blame] | 20 | public class RawDecoder { |
| 21 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 22 | public RawDecoder(Graphics g, RfbInputStream is) { |
| 23 | setGraphics(g); |
| 24 | setRfbInputStream(is); |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 25 | // FIXME: cm24 created in getColorModel24. |
| 26 | // Remove if no bugs |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 27 | cm24 = new DirectColorModel(24, 0xFF0000, 0x00FF00, 0x0000FF); |
| 28 | } |
| 29 | |
| 30 | public RawDecoder(Graphics g, RfbInputStream is, int frameBufferW, |
| 31 | int frameBufferH) { |
| 32 | setGraphics(g); |
| 33 | setRfbInputStream(is); |
| 34 | setFrameBufferSize(frameBufferW, frameBufferH); |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 35 | // FIXME: cm24 created in getColorModel24. |
| 36 | // Remove if no bugs |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 37 | cm24 = new DirectColorModel(24, 0xFF0000, 0x00FF00, 0x0000FF); |
| 38 | } |
| 39 | |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 40 | // |
| 41 | // Set methods to set value of non-static protected members of class |
| 42 | // |
| 43 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 44 | public void setRfbInputStream(RfbInputStream is) { |
| 45 | rfbis = is; |
| 46 | } |
| 47 | |
| 48 | public void setGraphics(Graphics g) { |
| 49 | graphics = g; |
| 50 | } |
| 51 | |
| 52 | public void setBPP(int bpp) { |
| 53 | bytesPerPixel = bpp; |
| 54 | } |
| 55 | |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 56 | public void setFrameBufferSize(int w, int h) { |
| 57 | framebufferWidth = w; |
| 58 | framebufferHeight = h; |
| 59 | } |
| 60 | |
enikey | 292cf9d | 2008-12-19 05:30:11 +0000 | [diff] [blame] | 61 | public void setSessionRecorder(RecordInterface ri) { |
| 62 | rec = ri; |
| 63 | } |
| 64 | |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 65 | // |
enikey | 3e9ff9f | 2008-12-24 03:30:29 +0000 | [diff] [blame] | 66 | // FIXME: Rename this method after we don't need RecordInterface |
| 67 | // in RawDecoder class to record session |
| 68 | // |
| 69 | |
enikey | c6f6bab | 2008-12-24 03:41:00 +0000 | [diff] [blame] | 70 | public void setDataOutputStream(DataOutput os) { |
enikey | 3e9ff9f | 2008-12-24 03:30:29 +0000 | [diff] [blame] | 71 | dos = os; |
| 72 | } |
| 73 | |
| 74 | // |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 75 | // FIXME: This method may be useless in future, remove if so |
| 76 | // |
| 77 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 78 | public int getBPP() { |
| 79 | return bytesPerPixel; |
| 80 | } |
| 81 | |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 82 | // |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 83 | // Decodes Raw Pixels data and draw it into graphics |
| 84 | // |
| 85 | |
| 86 | public void handleRect(int x, int y, int w, int h) throws IOException, Exception { |
| 87 | if (bytesPerPixel == 1) { |
| 88 | for (int dy = y; dy < y + h; dy++) { |
| 89 | if (pixels8 != null) { |
| 90 | rfbis.readFully(pixels8, dy * framebufferWidth + x, w); |
| 91 | } |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 92 | // |
enikey | 5276d79 | 2008-12-24 04:12:34 +0000 | [diff] [blame^] | 93 | // Save decoded data to record output stream |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 94 | // |
enikey | 5276d79 | 2008-12-24 04:12:34 +0000 | [diff] [blame^] | 95 | if (dos != null) { |
| 96 | dos.write(pixels8, dy * framebufferWidth + x, w); |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 97 | } |
| 98 | } |
| 99 | } else { |
| 100 | byte[] buf = new byte[w * 4]; |
| 101 | int i, offset; |
| 102 | for (int dy = y; dy < y + h; dy++) { |
| 103 | rfbis.readFully(buf); |
| 104 | // |
enikey | 5276d79 | 2008-12-24 04:12:34 +0000 | [diff] [blame^] | 105 | // Save decoded data to record output stream |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 106 | // |
enikey | 5276d79 | 2008-12-24 04:12:34 +0000 | [diff] [blame^] | 107 | if (dos != null) { |
| 108 | dos.write(buf); |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 109 | } |
| 110 | offset = dy * framebufferWidth + x; |
| 111 | if (pixels24 != null) { |
| 112 | for (i = 0; i < w; i++) { |
| 113 | pixels24[offset + i] = |
| 114 | (buf[i * 4 + 2] & 0xFF) << 16 | |
| 115 | (buf[i * 4 + 1] & 0xFF) << 8 | |
| 116 | (buf[i * 4] & 0xFF); |
| 117 | } //for |
| 118 | } // if |
| 119 | } // for |
| 120 | } // else |
| 121 | handleUpdatedPixels(x, y, w, h); |
| 122 | } // void |
enikey | 3e9ff9f | 2008-12-24 03:30:29 +0000 | [diff] [blame] | 123 | |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 124 | // |
| 125 | // Display newly updated area of pixels. |
| 126 | // |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 127 | |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 128 | protected void handleUpdatedPixels(int x, int y, int w, int h) { |
| 129 | // Draw updated pixels of the off-screen image. |
| 130 | pixelsSource.newPixels(x, y, w, h); |
| 131 | graphics.setClip(x, y, w, h); |
| 132 | graphics.drawImage(rawPixelsImage, 0, 0, null); |
| 133 | graphics.setClip(0, 0, framebufferWidth, framebufferHeight); |
| 134 | } |
| 135 | |
| 136 | // |
enikey | 663025d | 2008-12-19 05:14:15 +0000 | [diff] [blame] | 137 | // Updates pixels data. |
enikey | 4f92da4 | 2008-12-19 05:28:12 +0000 | [diff] [blame] | 138 | // This method must be called when framebuffer is resized |
enikey | 663025d | 2008-12-19 05:14:15 +0000 | [diff] [blame] | 139 | // or BPP is changed. |
| 140 | // |
| 141 | |
| 142 | public void update() { |
| 143 | // Images with raw pixels should be re-allocated on every change |
| 144 | // of geometry or pixel format. |
| 145 | int fbWidth = framebufferWidth; |
| 146 | int fbHeight = framebufferHeight; |
| 147 | |
| 148 | if (bytesPerPixel == 1) { |
| 149 | pixels24 = null; |
| 150 | pixels8 = new byte[fbWidth * fbHeight]; |
| 151 | pixelsSource = new MemoryImageSource(fbWidth, fbHeight, getColorModel8(), |
| 152 | pixels8, 0, fbWidth); |
| 153 | } else { |
| 154 | pixels8 = null; |
| 155 | pixels24 = new int[fbWidth * fbHeight]; |
| 156 | pixelsSource = |
| 157 | new MemoryImageSource(fbWidth, fbHeight, cm24, pixels24, 0, fbWidth); |
| 158 | } |
| 159 | pixelsSource.setAnimated(true); |
| 160 | rawPixelsImage = Toolkit.getDefaultToolkit().createImage(pixelsSource); |
| 161 | } |
| 162 | |
| 163 | // |
enikey | 1924c44 | 2008-12-19 05:36:36 +0000 | [diff] [blame] | 164 | // Private static members access methods |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 165 | // |
| 166 | |
| 167 | protected ColorModel getColorModel8() { |
| 168 | if (cm8 == null) { |
| 169 | cm8 = cm8 = new DirectColorModel(8, 7, (7 << 3), (3 << 6)); |
| 170 | } |
| 171 | return cm8; |
| 172 | } |
| 173 | |
| 174 | protected ColorModel getColorModel24() { |
| 175 | if (cm24 == null) { |
| 176 | cm24 = new DirectColorModel(24, 0xFF0000, 0x00FF00, 0x0000FF); |
| 177 | } |
| 178 | return cm24; |
| 179 | } |
| 180 | |
| 181 | protected Color[]getColor256() { |
| 182 | if (color256 == null) { |
| 183 | color256 = new Color[256]; |
| 184 | for (int i = 0; i < 256; i++) |
| 185 | color256[i] = new Color(cm8.getRGB(i)); |
| 186 | } |
| 187 | return color256; |
| 188 | } |
| 189 | |
| 190 | // |
| 191 | // Unique data for every decoder (? maybe not ?) |
| 192 | // |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 193 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 194 | protected int bytesPerPixel = 4; |
| 195 | protected int framebufferWidth = 0; |
| 196 | protected int framebufferHeight = 0; |
| 197 | protected RfbInputStream rfbis = null; |
| 198 | protected Graphics graphics = null; |
| 199 | protected RecordInterface rec = null; |
enikey | c6f6bab | 2008-12-24 03:41:00 +0000 | [diff] [blame] | 200 | protected DataOutput dos = null; |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 201 | |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 202 | // |
| 203 | // This data must be shared between decoders |
| 204 | // |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 205 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 206 | protected static byte []pixels8 = null; |
| 207 | protected static int []pixels24 = null; |
| 208 | protected static MemoryImageSource pixelsSource = null; |
| 209 | protected static Image rawPixelsImage = null; |
| 210 | |
enikey | b3e19f7 | 2008-12-19 04:54:49 +0000 | [diff] [blame] | 211 | // |
| 212 | // Access to this static members only though protected methods |
| 213 | // |
enikey | 2c23ba1 | 2008-12-19 05:34:17 +0000 | [diff] [blame] | 214 | |
enikey | 2f811b8 | 2008-12-19 04:45:15 +0000 | [diff] [blame] | 215 | private static ColorModel cm8 = null; |
| 216 | private static ColorModel cm24 = null; |
| 217 | private static Color []color256 = null; |
enikey | 00e0dbf | 2008-12-08 10:59:59 +0000 | [diff] [blame] | 218 | } |