mirror of
https://codeberg.org/Freeyourgadget/Gadgetbridge
synced 2025-01-15 12:17:33 +01:00
Move screenshot encoding to PebbleProtocol
This commit is contained in:
parent
55cecceb38
commit
531aef61fd
@ -17,9 +17,13 @@
|
||||
package nodomain.freeyourgadget.gadgetbridge.deviceevents;
|
||||
|
||||
public class GBDeviceEventScreenshot extends GBDeviceEvent {
|
||||
public int width;
|
||||
public int height;
|
||||
public byte bpp;
|
||||
public byte[] clut;
|
||||
public byte[] data;
|
||||
private final byte[] data;
|
||||
|
||||
public GBDeviceEventScreenshot(final byte[] data) {
|
||||
this.data = data;
|
||||
}
|
||||
|
||||
public byte[] getData() {
|
||||
return data;
|
||||
}
|
||||
}
|
||||
|
@ -379,6 +379,11 @@ public abstract class AbstractDeviceSupport implements DeviceSupport {
|
||||
}
|
||||
|
||||
private void handleGBDeviceEvent(GBDeviceEventScreenshot screenshot) {
|
||||
if (screenshot.getData() == null) {
|
||||
LOG.warn("Screnshot data is null");
|
||||
return;
|
||||
}
|
||||
|
||||
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMdd-hhmmss", Locale.US);
|
||||
String filename = "screenshot_" + dateFormat.format(new Date()) + ".bmp";
|
||||
|
||||
|
@ -27,6 +27,10 @@ import org.json.JSONObject;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.ByteOrder;
|
||||
import java.util.ArrayList;
|
||||
@ -65,6 +69,7 @@ import nodomain.freeyourgadget.gadgetbridge.model.NotificationType;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.Weather;
|
||||
import nodomain.freeyourgadget.gadgetbridge.model.WeatherSpec;
|
||||
import nodomain.freeyourgadget.gadgetbridge.service.serial.GBDeviceProtocol;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.FileUtils;
|
||||
import nodomain.freeyourgadget.gadgetbridge.util.GB;
|
||||
|
||||
public class PebbleProtocol extends GBDeviceProtocol {
|
||||
@ -281,7 +286,12 @@ public class PebbleProtocol extends GBDeviceProtocol {
|
||||
boolean mEnablePebbleKit = false;
|
||||
boolean mAlwaysACKPebbleKit = false;
|
||||
private boolean mForceProtocol = false;
|
||||
private GBDeviceEventScreenshot mDevEventScreenshot = null;
|
||||
|
||||
private byte[] screenshotData = null;
|
||||
private int screenshotWidth;
|
||||
private int screenshotHeight;
|
||||
private byte screenshotBpp;
|
||||
private byte[] screenshotClut;
|
||||
private int mScreenshotRemaining = -1;
|
||||
|
||||
//monochrome black + white
|
||||
@ -1945,27 +1955,26 @@ public class PebbleProtocol extends GBDeviceProtocol {
|
||||
}
|
||||
|
||||
private GBDeviceEventScreenshot decodeScreenshot(ByteBuffer buf, int length) {
|
||||
if (mDevEventScreenshot == null) {
|
||||
if (screenshotData == null) {
|
||||
byte result = buf.get();
|
||||
mDevEventScreenshot = new GBDeviceEventScreenshot();
|
||||
int version = buf.getInt();
|
||||
if (result != 0) {
|
||||
return null;
|
||||
}
|
||||
mDevEventScreenshot.width = buf.getInt();
|
||||
mDevEventScreenshot.height = buf.getInt();
|
||||
screenshotWidth = buf.getInt();
|
||||
screenshotHeight = buf.getInt();
|
||||
|
||||
if (version == 1) {
|
||||
mDevEventScreenshot.bpp = 1;
|
||||
mDevEventScreenshot.clut = clut_pebble;
|
||||
screenshotBpp = 1;
|
||||
screenshotClut = clut_pebble;
|
||||
} else {
|
||||
mDevEventScreenshot.bpp = 8;
|
||||
mDevEventScreenshot.clut = clut_pebbletime;
|
||||
screenshotBpp = 8;
|
||||
screenshotClut = clut_pebbletime;
|
||||
}
|
||||
|
||||
mScreenshotRemaining = (mDevEventScreenshot.width * mDevEventScreenshot.height * mDevEventScreenshot.bpp) / 8;
|
||||
mScreenshotRemaining = (screenshotWidth * screenshotHeight * screenshotBpp) / 8;
|
||||
|
||||
mDevEventScreenshot.data = new byte[mScreenshotRemaining];
|
||||
screenshotData = new byte[mScreenshotRemaining];
|
||||
length -= 13;
|
||||
}
|
||||
if (mScreenshotRemaining == -1) {
|
||||
@ -1973,26 +1982,68 @@ public class PebbleProtocol extends GBDeviceProtocol {
|
||||
}
|
||||
for (int i = 0; i < length; i++) {
|
||||
byte corrected = buf.get();
|
||||
if (mDevEventScreenshot.bpp == 1) {
|
||||
if (screenshotBpp == 1) {
|
||||
corrected = reverseBits(corrected);
|
||||
} else {
|
||||
corrected = (byte) (corrected & 0b00111111);
|
||||
}
|
||||
|
||||
mDevEventScreenshot.data[mDevEventScreenshot.data.length - mScreenshotRemaining + i] = corrected;
|
||||
screenshotData[screenshotData.length - mScreenshotRemaining + i] = corrected;
|
||||
}
|
||||
mScreenshotRemaining -= length;
|
||||
LOG.info("Screenshot remaining bytes " + mScreenshotRemaining);
|
||||
if (mScreenshotRemaining == 0) {
|
||||
mScreenshotRemaining = -1;
|
||||
LOG.info("Got screenshot : " + mDevEventScreenshot.width + "x" + mDevEventScreenshot.height + " " + "pixels");
|
||||
GBDeviceEventScreenshot devEventScreenshot = mDevEventScreenshot;
|
||||
mDevEventScreenshot = null;
|
||||
LOG.info("Got screenshot : " + screenshotWidth + "x" + screenshotHeight + " " + "pixels");
|
||||
GBDeviceEventScreenshot devEventScreenshot = new GBDeviceEventScreenshot(encodeScreenshotBmp());
|
||||
screenshotData = null;
|
||||
return devEventScreenshot;
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
private byte[] encodeScreenshotBmp() {
|
||||
final int FILE_HEADER_SIZE = 14;
|
||||
final int INFO_HEADER_SIZE = 40;
|
||||
|
||||
try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
|
||||
ByteBuffer headerbuf = ByteBuffer.allocate(FILE_HEADER_SIZE + INFO_HEADER_SIZE + screenshotClut.length);
|
||||
headerbuf.order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
// file header
|
||||
headerbuf.put((byte) 'B');
|
||||
headerbuf.put((byte) 'M');
|
||||
headerbuf.putInt(0); // size in bytes (uncompressed = 0)
|
||||
headerbuf.putInt(0); // reserved
|
||||
headerbuf.putInt(FILE_HEADER_SIZE + INFO_HEADER_SIZE + screenshotClut.length);
|
||||
|
||||
// info header
|
||||
headerbuf.putInt(INFO_HEADER_SIZE);
|
||||
headerbuf.putInt(screenshotWidth);
|
||||
headerbuf.putInt(-screenshotHeight);
|
||||
headerbuf.putShort((short) 1); // planes
|
||||
headerbuf.putShort((short) screenshotBpp);
|
||||
headerbuf.putInt(0); // compression
|
||||
headerbuf.putInt(0); // length of pixeldata in bytes (uncompressed=0)
|
||||
headerbuf.putInt(0); // pixels per meter (x)
|
||||
headerbuf.putInt(0); // pixels per meter (y)
|
||||
headerbuf.putInt(screenshotClut.length / 4); // number of colors in CLUT
|
||||
headerbuf.putInt(0); // numbers of used colors
|
||||
headerbuf.put(screenshotClut);
|
||||
baos.write(headerbuf.array());
|
||||
int rowbytes = (screenshotWidth * screenshotBpp) / 8;
|
||||
byte[] pad = new byte[rowbytes % 4];
|
||||
for (int i = 0; i < screenshotHeight; i++) {
|
||||
baos.write(screenshotData, rowbytes * i, rowbytes);
|
||||
baos.write(pad);
|
||||
}
|
||||
return baos.toByteArray();
|
||||
} catch (final IOException e) {
|
||||
LOG.warn("Failed to encode screenshot to bpm");
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private GBDeviceEvent[] decodeAction(ByteBuffer buf) {
|
||||
buf.order(ByteOrder.LITTLE_ENDIAN);
|
||||
byte command = buf.get();
|
||||
|
@ -347,44 +347,12 @@ public class GB {
|
||||
}
|
||||
|
||||
public static String writeScreenshot(GBDeviceEventScreenshot screenshot, String filename) throws IOException {
|
||||
LOG.info("Will write screenshot as {}", filename);
|
||||
|
||||
LOG.info("Will write screenshot: " + screenshot.width + "x" + screenshot.height + "x" + screenshot.bpp + "bpp");
|
||||
final int FILE_HEADER_SIZE = 14;
|
||||
final int INFO_HEADER_SIZE = 40;
|
||||
|
||||
File dir = FileUtils.getExternalFilesDir();
|
||||
File outputFile = new File(dir, filename);
|
||||
final File dir = FileUtils.getExternalFilesDir();
|
||||
final File outputFile = new File(dir, filename);
|
||||
try (FileOutputStream fos = new FileOutputStream(outputFile)) {
|
||||
ByteBuffer headerbuf = ByteBuffer.allocate(FILE_HEADER_SIZE + INFO_HEADER_SIZE + screenshot.clut.length);
|
||||
headerbuf.order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
// file header
|
||||
headerbuf.put((byte) 'B');
|
||||
headerbuf.put((byte) 'M');
|
||||
headerbuf.putInt(0); // size in bytes (uncompressed = 0)
|
||||
headerbuf.putInt(0); // reserved
|
||||
headerbuf.putInt(FILE_HEADER_SIZE + INFO_HEADER_SIZE + screenshot.clut.length);
|
||||
|
||||
// info header
|
||||
headerbuf.putInt(INFO_HEADER_SIZE);
|
||||
headerbuf.putInt(screenshot.width);
|
||||
headerbuf.putInt(-screenshot.height);
|
||||
headerbuf.putShort((short) 1); // planes
|
||||
headerbuf.putShort((short) screenshot.bpp);
|
||||
headerbuf.putInt(0); // compression
|
||||
headerbuf.putInt(0); // length of pixeldata in bytes (uncompressed=0)
|
||||
headerbuf.putInt(0); // pixels per meter (x)
|
||||
headerbuf.putInt(0); // pixels per meter (y)
|
||||
headerbuf.putInt(screenshot.clut.length / 4); // number of colors in CLUT
|
||||
headerbuf.putInt(0); // numbers of used colors
|
||||
headerbuf.put(screenshot.clut);
|
||||
fos.write(headerbuf.array());
|
||||
int rowbytes = (screenshot.width * screenshot.bpp) / 8;
|
||||
byte[] pad = new byte[rowbytes % 4];
|
||||
for (int i = 0; i < screenshot.height; i++) {
|
||||
fos.write(screenshot.data, rowbytes * i, rowbytes);
|
||||
fos.write(pad);
|
||||
}
|
||||
fos.write(screenshot.getData());
|
||||
}
|
||||
return outputFile.getAbsolutePath();
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user