2016-04-09 00:39:53 +02:00
|
|
|
|
import java.awt.Graphics2D;
|
|
|
|
|
import java.awt.image.BufferedImage;
|
2016-07-11 20:15:35 +02:00
|
|
|
|
import java.io.BufferedReader;
|
2016-04-09 00:39:53 +02:00
|
|
|
|
import java.io.File;
|
|
|
|
|
import java.io.IOException;
|
2016-07-11 20:15:35 +02:00
|
|
|
|
import java.io.InputStreamReader;
|
2016-04-09 00:39:53 +02:00
|
|
|
|
import java.net.URL;
|
|
|
|
|
import java.util.Arrays;
|
2016-06-29 01:27:40 +02:00
|
|
|
|
import java.util.regex.Matcher;
|
2016-04-09 00:39:53 +02:00
|
|
|
|
|
|
|
|
|
import javax.imageio.ImageIO;
|
|
|
|
|
|
2016-06-23 14:56:57 +02:00
|
|
|
|
/**
|
|
|
|
|
* Simple program to print images to the shell using 24 bit ANSI color codes and Unicode block graphics characters.
|
|
|
|
|
*
|
|
|
|
|
* License: Apache 2.0
|
|
|
|
|
* @author Stefan Haustein
|
|
|
|
|
*/
|
2016-04-09 00:39:53 +02:00
|
|
|
|
public class TerminalImageViewer {
|
|
|
|
|
|
2016-07-11 20:15:35 +02:00
|
|
|
|
static boolean grayscale = false;
|
|
|
|
|
static int mode = Ansi.MODE_24BIT;
|
2016-07-17 03:54:48 +02:00
|
|
|
|
static boolean html = false;
|
2016-07-11 20:15:35 +02:00
|
|
|
|
|
|
|
|
|
|
2016-04-09 00:39:53 +02:00
|
|
|
|
/**
|
|
|
|
|
* Main method, handles command line arguments and loads and scales images.
|
|
|
|
|
*/
|
|
|
|
|
public static void main(String[] args) throws IOException {
|
|
|
|
|
if (args.length == 0) {
|
2016-06-29 01:27:40 +02:00
|
|
|
|
System.out.println(
|
2016-10-01 13:13:03 +02:00
|
|
|
|
"Image file name required.\n\n" +
|
|
|
|
|
" - Use -w and -h to set the maximum width and height in characters (defaults: 80, 24).\n" +
|
|
|
|
|
" - Use -256 for 256 color mode, -grayscale for grayscale and -stdin to obtain file names from stdin.\n" +
|
|
|
|
|
" - When multiple files are supplied, -c sets the number of images per row (default: 4).");
|
2016-04-09 00:39:53 +02:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-09 10:54:10 +02:00
|
|
|
|
int start = 0;
|
2016-06-29 01:27:40 +02:00
|
|
|
|
int maxWidth = 80;
|
|
|
|
|
int maxHeight = 24;
|
2016-10-01 13:13:03 +02:00
|
|
|
|
int columns = 4;
|
2016-07-11 20:15:35 +02:00
|
|
|
|
boolean stdin = false;
|
2016-06-26 13:09:25 +02:00
|
|
|
|
while (start < args.length && args[start].startsWith("-")) {
|
|
|
|
|
String option = args[start];
|
|
|
|
|
if (option.equals("-w") && args.length > start + 1) {
|
2016-06-29 01:27:40 +02:00
|
|
|
|
maxWidth = Integer.parseInt(args[++start]);
|
|
|
|
|
} else if (option.equals("-h") && args.length > start + 1) {
|
2016-10-01 13:13:03 +02:00
|
|
|
|
maxHeight = Integer.parseInt(args[++start]);
|
|
|
|
|
} else if (option.equals("-c") && args.length > start + 1) {
|
|
|
|
|
columns = Integer.parseInt(args[++start]);
|
2016-06-26 13:09:25 +02:00
|
|
|
|
} else if (option.equals("-256")) {
|
2016-06-29 01:27:40 +02:00
|
|
|
|
mode = (mode & ~Ansi.MODE_24BIT) | Ansi.MODE_256;
|
|
|
|
|
} else if (option.equals("-grayscale")) {
|
|
|
|
|
grayscale = true;
|
2016-07-17 03:54:48 +02:00
|
|
|
|
} else if (option.equals("-html")) {
|
|
|
|
|
html = true;
|
2016-07-11 20:15:35 +02:00
|
|
|
|
} else if (option.equals("-stdin")) {
|
|
|
|
|
stdin = true;
|
2016-06-26 13:09:25 +02:00
|
|
|
|
}
|
|
|
|
|
start++;
|
2016-04-09 10:54:10 +02:00
|
|
|
|
}
|
2016-04-09 00:39:53 +02:00
|
|
|
|
|
2016-06-29 01:27:40 +02:00
|
|
|
|
maxWidth *= 4;
|
|
|
|
|
maxHeight *= 8;
|
|
|
|
|
|
2016-07-11 20:15:35 +02:00
|
|
|
|
if (stdin) {
|
|
|
|
|
BufferedReader reader = new BufferedReader(new InputStreamReader(System.in));
|
|
|
|
|
while (true) {
|
|
|
|
|
String name = reader.readLine();
|
|
|
|
|
if (name == null || name.isEmpty()) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
convert(name, maxWidth, maxHeight);
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
2016-07-11 20:15:35 +02:00
|
|
|
|
} else if (start == args.length - 1 && (isUrl(args[start]) || !new File(args[start]).isDirectory())) {
|
|
|
|
|
convert(args[start], maxWidth, maxHeight);
|
2016-04-09 10:54:10 +02:00
|
|
|
|
} else {
|
|
|
|
|
// Directory-style rendering.
|
|
|
|
|
int index = 0;
|
2016-10-01 13:13:03 +02:00
|
|
|
|
int cw = (maxWidth - 2 * (columns - 1) * 4) / (4 * columns);
|
2016-04-09 10:54:10 +02:00
|
|
|
|
int tw = cw * 4;
|
2016-06-29 01:27:40 +02:00
|
|
|
|
|
2016-04-09 10:54:10 +02:00
|
|
|
|
while (index < args.length) {
|
2016-10-01 13:13:03 +02:00
|
|
|
|
BufferedImage image = new BufferedImage(tw * columns + 24, tw, grayscale ? BufferedImage.TYPE_BYTE_GRAY : BufferedImage.TYPE_INT_RGB);
|
2016-04-09 10:54:10 +02:00
|
|
|
|
Graphics2D graphics = image.createGraphics();
|
|
|
|
|
int count = 0;
|
|
|
|
|
StringBuilder sb = new StringBuilder();
|
2016-10-01 13:13:03 +02:00
|
|
|
|
while (index < args.length && count < columns) {
|
2016-04-09 10:54:10 +02:00
|
|
|
|
String name = args[index++];
|
|
|
|
|
try {
|
|
|
|
|
BufferedImage original = loadImage(name);
|
|
|
|
|
int cut = name.lastIndexOf('/');
|
|
|
|
|
sb.append(name.substring(cut + 1));
|
|
|
|
|
int th = original.getHeight() * tw / original.getWidth();
|
|
|
|
|
graphics.drawImage(original, count * (tw + 8), (tw - th) / 2, tw, th, null);
|
|
|
|
|
count++;
|
|
|
|
|
int sl = count * (cw + 2);
|
|
|
|
|
while (sb.length() < sl - 2) {
|
|
|
|
|
sb.append(' ');
|
|
|
|
|
}
|
|
|
|
|
sb.setLength(sl - 2);
|
|
|
|
|
sb.append(" ");
|
|
|
|
|
} catch (Exception e) {
|
|
|
|
|
// Probably no image; ignore.
|
|
|
|
|
}
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
2016-06-26 13:09:25 +02:00
|
|
|
|
dump(image, mode);
|
2016-04-09 10:54:10 +02:00
|
|
|
|
System.out.println(sb.toString());
|
|
|
|
|
System.out.println();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-29 01:27:40 +02:00
|
|
|
|
static boolean isUrl(String name) {
|
|
|
|
|
return name.startsWith("http://") || name.startsWith("https://");
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-11 20:15:35 +02:00
|
|
|
|
static void convert(String name, int maxWidth, int maxHeight) throws IOException {
|
|
|
|
|
BufferedImage original = loadImage(name);
|
|
|
|
|
|
|
|
|
|
float originalWidth = original.getWidth();
|
|
|
|
|
float originalHeight = original.getHeight();
|
|
|
|
|
float scale = Math.min(maxWidth / originalWidth, maxHeight / originalHeight);
|
|
|
|
|
int height = (int) (originalHeight * scale);
|
|
|
|
|
int width = (int) (originalWidth * scale);
|
|
|
|
|
|
|
|
|
|
if (originalWidth == width && !grayscale) {
|
|
|
|
|
dump(original, mode);
|
|
|
|
|
} else {
|
|
|
|
|
BufferedImage image = new BufferedImage(width, height, grayscale ? BufferedImage.TYPE_BYTE_GRAY : BufferedImage.TYPE_INT_RGB);
|
|
|
|
|
Graphics2D graphics = image.createGraphics();
|
|
|
|
|
graphics.drawImage(original, 0, 0, width, height, null);
|
|
|
|
|
dump(image, mode);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-09 10:54:10 +02:00
|
|
|
|
static BufferedImage loadImage(String name) throws IOException {
|
2016-06-29 01:27:40 +02:00
|
|
|
|
if (isUrl(name)) {
|
2016-04-09 10:54:10 +02:00
|
|
|
|
URL url = new URL(name);
|
|
|
|
|
return ImageIO.read(url);
|
|
|
|
|
}
|
|
|
|
|
return ImageIO.read(new File(name));
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-26 13:09:25 +02:00
|
|
|
|
static void dump(BufferedImage image, int mode) {
|
2016-04-09 10:54:10 +02:00
|
|
|
|
int w = image.getWidth();
|
|
|
|
|
ImageData imageData = new ImageData(w, image.getHeight());
|
|
|
|
|
byte[] data = imageData.data;
|
|
|
|
|
int[] rgbArray = new int[w];
|
|
|
|
|
for (int y = 0; y < image.getHeight(); y++) {
|
|
|
|
|
image.getRGB(0, y, image.getWidth(), 1, rgbArray, 0, w);
|
|
|
|
|
int pos = y * w * 4;
|
|
|
|
|
for (int x = 0; x < w; x++) {
|
|
|
|
|
int rgb = rgbArray[x];
|
|
|
|
|
data[pos++] = (byte) (rgb >> 16);
|
|
|
|
|
data[pos++] = (byte) (rgb >> 8);
|
|
|
|
|
data[pos++] = (byte) rgb;
|
|
|
|
|
pos++;
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-06-26 13:09:25 +02:00
|
|
|
|
System.out.print(imageData.dump(mode));
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-17 03:40:53 +02:00
|
|
|
|
|
|
|
|
|
|
2016-04-09 00:39:53 +02:00
|
|
|
|
/**
|
|
|
|
|
* ANSI control code helpers
|
|
|
|
|
*/
|
|
|
|
|
static class Ansi {
|
|
|
|
|
public static final String RESET = "\u001b[0m";
|
2016-06-29 01:27:40 +02:00
|
|
|
|
public static int FG = 1;
|
|
|
|
|
public static int BG = 2;
|
|
|
|
|
public static int MODE_256 = 4;
|
|
|
|
|
public static int MODE_24BIT = 8;
|
|
|
|
|
|
|
|
|
|
public static final int[] COLOR_STEPS = {0, 0x5f, 0x87, 0xaf, 0xd7, 0xff};
|
|
|
|
|
public static final int[] GRAYSCALE = {0x08, 0x12, 0x1c, 0x26, 0x30, 0x3a, 0x44, 0x4e, 0x58, 0x62, 0x6c, 0x76,
|
|
|
|
|
0x80, 0x8a, 0x94, 0x9e, 0xa8, 0xb2, 0xbc, 0xc6, 0xd0, 0xda, 0xe4, 0xee};
|
|
|
|
|
|
|
|
|
|
static int bestIndex(int v, int[] options) {
|
|
|
|
|
int index = Arrays.binarySearch(options, v);
|
|
|
|
|
if (index < 0) {
|
|
|
|
|
index = -index - 1;
|
|
|
|
|
// need to check [index] and [index - 1]
|
|
|
|
|
if (index == options.length) {
|
|
|
|
|
index = options.length - 1;
|
|
|
|
|
} else if (index > 0) {
|
|
|
|
|
int val0 = options[index - 1];
|
|
|
|
|
int val1 = options[index];
|
|
|
|
|
if (v - val0 < val1 - v) {
|
|
|
|
|
index = index - 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return index;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int sqr(int i) {
|
|
|
|
|
return i * i;
|
|
|
|
|
}
|
2016-04-09 00:39:53 +02:00
|
|
|
|
|
2016-06-26 13:09:25 +02:00
|
|
|
|
public static int clamp(int value, int min, int max) {
|
|
|
|
|
return Math.min(Math.max(value, min), max);
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-06-26 13:09:25 +02:00
|
|
|
|
public static String color(int flags, int r, int g, int b) {
|
|
|
|
|
r = clamp(r, 0, 255);
|
|
|
|
|
g = clamp(g, 0, 255);
|
|
|
|
|
b = clamp(b, 0, 255);
|
|
|
|
|
|
|
|
|
|
boolean bg = (flags & BG) != 0;
|
|
|
|
|
|
2016-06-29 01:27:40 +02:00
|
|
|
|
if ((flags & MODE_256) == 0) {
|
2016-06-26 13:09:25 +02:00
|
|
|
|
return (bg ? "\u001b[48;2;" : "\u001b[38;2;") + r + ";" + g + ";" + b + "m";
|
|
|
|
|
}
|
2016-06-29 01:27:40 +02:00
|
|
|
|
int rIdx = bestIndex(r, COLOR_STEPS);
|
|
|
|
|
int gIdx = bestIndex(g, COLOR_STEPS);
|
|
|
|
|
int bIdx = bestIndex(b, COLOR_STEPS);
|
|
|
|
|
|
|
|
|
|
int rQ = COLOR_STEPS[rIdx];
|
|
|
|
|
int gQ = COLOR_STEPS[gIdx];
|
|
|
|
|
int bQ = COLOR_STEPS[bIdx];
|
|
|
|
|
|
|
|
|
|
int gray = Math.round(r * 0.2989f + g * 0.5870f + b * 0.1140f);
|
|
|
|
|
|
|
|
|
|
int grayIdx = bestIndex(gray, GRAYSCALE);
|
|
|
|
|
int grayQ = GRAYSCALE[grayIdx];
|
|
|
|
|
|
|
|
|
|
int colorIndex;
|
|
|
|
|
if (0.3 * sqr(rQ-r) + 0.59 * sqr(gQ-g) + 0.11 *sqr(bQ-b) <
|
|
|
|
|
0.3 * sqr(grayQ-r) + 0.59 * sqr(grayQ-g) + 0.11 * sqr(grayQ-b)) {
|
|
|
|
|
colorIndex = 16 + 36 * rIdx + 6 * gIdx + bIdx;
|
|
|
|
|
} else {
|
|
|
|
|
colorIndex = 232 + grayIdx; // 1..24 -> 232..255
|
|
|
|
|
}
|
|
|
|
|
return (bg ? "\u001B[48;5;" : "\u001B[38;5;") + colorIndex + "m";
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2016-04-09 01:14:19 +02:00
|
|
|
|
* Converts 4x8 RGB pixel to a unicode character and a foreground and background color:
|
2016-04-09 00:39:53 +02:00
|
|
|
|
* Uses a variation of the median cut algorithm to determine a two-color palette for the
|
|
|
|
|
* character, then creates a corresponding bitmap for the partial image covered by the
|
|
|
|
|
* character and finds the best match in the character bitmap table.
|
|
|
|
|
*/
|
|
|
|
|
static class BlockChar {
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Assumed bitmaps of the supported characters
|
|
|
|
|
*/
|
|
|
|
|
static int[] BITMAPS = new int[] {
|
2016-07-17 03:54:48 +02:00
|
|
|
|
0x00000000, '\u00a0',
|
2016-04-09 00:39:53 +02:00
|
|
|
|
|
2016-04-09 17:15:33 +02:00
|
|
|
|
// Block graphics
|
|
|
|
|
|
|
|
|
|
// 0xffff0000, '\u2580', // upper 1/2; redundant with inverse lower 1/2
|
2016-04-09 00:39:53 +02:00
|
|
|
|
|
|
|
|
|
0x0000000f, '\u2581', // lower 1/8
|
|
|
|
|
0x000000ff, '\u2582', // lower 1/4
|
|
|
|
|
0x00000fff, '\u2583',
|
|
|
|
|
0x0000ffff, '\u2584', // lower 1/2
|
|
|
|
|
0x000fffff, '\u2585',
|
|
|
|
|
0x00ffffff, '\u2586', // lower 3/4
|
|
|
|
|
0x0fffffff, '\u2587',
|
2016-04-09 17:15:33 +02:00
|
|
|
|
// 0xffffffff, '\u2588', // full; redundant with inverse space
|
2016-04-09 00:39:53 +02:00
|
|
|
|
|
|
|
|
|
0xeeeeeeee, '\u258a', // left 3/4
|
|
|
|
|
0xcccccccc, '\u258c', // left 1/2
|
|
|
|
|
0x88888888, '\u258e', // left 1/4
|
|
|
|
|
|
|
|
|
|
0x0000cccc, '\u2596', // quadrant lower left
|
|
|
|
|
0x00003333, '\u2597', // quadrant lower right
|
|
|
|
|
0xcccc0000, '\u2598', // quadrant upper left
|
2016-04-09 17:15:33 +02:00
|
|
|
|
// 0xccccffff, '\u2599', // 3/4 redundant with inverse 1/4
|
|
|
|
|
0xcccc3333, '\u259a', // diagonal 1/2
|
|
|
|
|
// 0xffffcccc, '\u259b', // 3/4 redundant
|
|
|
|
|
// 0xffff3333, '\u259c', // 3/4 redundant
|
|
|
|
|
0x33330000, '\u259d', // quadrant upper right
|
|
|
|
|
// 0x3333cccc, '\u259e', // 3/4 redundant
|
|
|
|
|
// 0x3333ffff, '\u259f', // 3/4 redundant
|
|
|
|
|
|
|
|
|
|
// Line drawing subset: no double lines, no complex light lines
|
|
|
|
|
// Simple light lines duplicated because there is no center pixel int the 4x8 matrix
|
|
|
|
|
|
|
|
|
|
0x000ff000, '\u2501', // Heavy horizontal
|
2016-04-09 16:05:31 +02:00
|
|
|
|
0x66666666, '\u2503', // Heavy vertical
|
|
|
|
|
|
|
|
|
|
0x00077666, '\u250f', // Heavy down and right
|
|
|
|
|
0x000ee666, '\u2513', // Heavy down and left
|
|
|
|
|
0x66677000, '\u2517', // Heavy up and right
|
|
|
|
|
0x666ee000, '\u251b', // Heavy up and left
|
|
|
|
|
|
|
|
|
|
0x66677666, '\u2523', // Heavy vertical and right
|
|
|
|
|
0x666ee666, '\u252b', // Heavy vertical and left
|
|
|
|
|
0x000ff666, '\u2533', // Heavy down and horizontal
|
|
|
|
|
0x666ff000, '\u253b', // Heavy up and horizontal
|
|
|
|
|
0x666ff666, '\u254b', // Heavy cross
|
|
|
|
|
|
|
|
|
|
0x000cc000, '\u2578', // Bold horizontal left
|
|
|
|
|
0x00066000, '\u2579', // Bold horizontal up
|
|
|
|
|
0x00033000, '\u257a', // Bold horizontal right
|
|
|
|
|
0x00066000, '\u257b', // Bold horizontal down
|
|
|
|
|
|
2017-04-06 21:26:37 +02:00
|
|
|
|
0x06600660, '\u254f', // Heavy double dash vertical
|
2016-04-09 16:05:31 +02:00
|
|
|
|
|
|
|
|
|
0x000f0000, '\u2500', // Light horizontal
|
|
|
|
|
0x0000f000, '\u2500', //
|
|
|
|
|
0x44444444, '\u2502', // Light vertical
|
|
|
|
|
0x22222222, '\u2502',
|
|
|
|
|
|
|
|
|
|
0x000e0000, '\u2574', // light left
|
|
|
|
|
0x0000e000, '\u2574', // light left
|
|
|
|
|
0x44440000, '\u2575', // light up
|
|
|
|
|
0x22220000, '\u2575', // light up
|
|
|
|
|
0x00030000, '\u2576', // light right
|
|
|
|
|
0x00003000, '\u2576', // light right
|
|
|
|
|
0x00004444, '\u2575', // light down
|
|
|
|
|
0x00002222, '\u2575', // light down
|
2016-04-09 17:15:33 +02:00
|
|
|
|
|
|
|
|
|
// Misc technical
|
|
|
|
|
|
2016-04-09 16:05:31 +02:00
|
|
|
|
0x44444444, '\u23a2', // [ extension
|
|
|
|
|
0x22222222, '\u23a5', // ] extension
|
|
|
|
|
|
|
|
|
|
//12345678
|
2016-04-09 17:15:33 +02:00
|
|
|
|
0x0f000000, '\u23ba', // Horizontal scanline 1
|
|
|
|
|
0x00f00000, '\u23bb', // Horizontal scanline 3
|
|
|
|
|
0x00000f00, '\u23bc', // Horizontal scanline 7
|
|
|
|
|
0x000000f0, '\u23bd', // Horizontal scanline 9
|
|
|
|
|
|
|
|
|
|
// Geometrical shapes. Tricky because some of them are too wide.
|
2016-04-09 16:05:31 +02:00
|
|
|
|
|
2016-04-09 17:15:33 +02:00
|
|
|
|
// 0x00ffff00, '\u25fe', // Black medium small square
|
2016-04-09 16:05:31 +02:00
|
|
|
|
0x00066000, '\u25aa', // Black small square
|
|
|
|
|
|
|
|
|
|
/*
|
2016-04-09 17:15:33 +02:00
|
|
|
|
0x11224488, '\u2571', // diagonals
|
|
|
|
|
0x88442211, '\u2572',
|
|
|
|
|
0x99666699, '\u2573',
|
|
|
|
|
|
2016-04-09 00:39:53 +02:00
|
|
|
|
0x000137f0, '\u25e2', // Triangles
|
|
|
|
|
0x0008cef0, '\u25e3',
|
|
|
|
|
0x000fec80, '\u25e4',
|
|
|
|
|
0x000f7310, '\u25e5'
|
2016-04-09 16:05:31 +02:00
|
|
|
|
*/
|
2016-04-09 00:39:53 +02:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/** Minimum value for each color channel. */
|
|
|
|
|
int[] min = new int[3];
|
|
|
|
|
|
|
|
|
|
/** Maximum value for each color channel. */
|
|
|
|
|
int[] max = new int[3];
|
|
|
|
|
|
|
|
|
|
/** Red, green and blue components of the selected background color. */
|
|
|
|
|
int[] bgColor = new int[3];
|
|
|
|
|
|
|
|
|
|
/** Red, green and blue components of the selected background color. */
|
|
|
|
|
int[] fgColor = new int[3];
|
|
|
|
|
|
|
|
|
|
/** The selected character. */
|
|
|
|
|
char character;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Converts a set of pixels to a unicode character and a background and foreground color.
|
|
|
|
|
* data contains the rgba values, p0 is the start point in data and scanWidth the number
|
|
|
|
|
* of bytes in each row of data.
|
|
|
|
|
*/
|
|
|
|
|
void load(byte[] data, int p0, int scanWidth) {
|
|
|
|
|
Arrays.fill(min, 255);
|
|
|
|
|
Arrays.fill(max, 0);
|
|
|
|
|
Arrays.fill(bgColor, 0);
|
|
|
|
|
Arrays.fill(fgColor, 0);
|
|
|
|
|
|
|
|
|
|
// Determine the minimum and maximum value for each color channel
|
|
|
|
|
int pos = p0;
|
|
|
|
|
for (int y = 0; y < 8; y++) {
|
|
|
|
|
for (int x = 0; x < 4; x++) {
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
|
int d = data[pos++] & 255;
|
|
|
|
|
min[i] = Math.min(min[i], d);
|
|
|
|
|
max[i] = Math.max(max[i], d);
|
|
|
|
|
}
|
|
|
|
|
pos++; // Alpha
|
|
|
|
|
}
|
|
|
|
|
pos += scanWidth - 16;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Determine the color channel with the greatest range.
|
|
|
|
|
int splitIndex = 0;
|
|
|
|
|
int bestSplit = 0;
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
|
if (max[i] - min[i] > bestSplit) {
|
|
|
|
|
bestSplit = max[i] - min[i];
|
|
|
|
|
splitIndex = i;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// We just split at the middle of the interval instead of computing the median.
|
|
|
|
|
int splitValue = min[splitIndex] + bestSplit / 2;
|
|
|
|
|
|
|
|
|
|
// Compute a bitmap using the given split and sum the color values for both buckets.
|
|
|
|
|
int bits = 0;
|
|
|
|
|
int fgCount = 0;
|
|
|
|
|
int bgCount = 0;
|
|
|
|
|
|
|
|
|
|
pos = p0;
|
|
|
|
|
for (int y = 0; y < 8; y++) {
|
|
|
|
|
for (int x = 0; x < 4; x++) {
|
|
|
|
|
bits = bits << 1;
|
|
|
|
|
int[] avg;
|
|
|
|
|
if ((data[pos + splitIndex] & 255) > splitValue) {
|
|
|
|
|
avg = fgColor;
|
|
|
|
|
bits |= 1;
|
|
|
|
|
fgCount++;
|
|
|
|
|
} else {
|
|
|
|
|
avg = bgColor;
|
|
|
|
|
bgCount++;
|
|
|
|
|
}
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
|
avg[i] += data[pos++] & 255;
|
|
|
|
|
}
|
|
|
|
|
pos++; // Alpha
|
|
|
|
|
}
|
|
|
|
|
pos += scanWidth - 16;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Calculate the average color value for each bucket
|
|
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
|
if (bgCount != 0) {
|
|
|
|
|
bgColor[i] /= bgCount;
|
|
|
|
|
}
|
|
|
|
|
if (fgCount != 0) {
|
|
|
|
|
fgColor[i] /= fgCount;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Find the best bitmap match by counting the bits that don't match, including
|
|
|
|
|
// the inverted bitmaps.
|
|
|
|
|
int bestDiff = Integer.MAX_VALUE;
|
|
|
|
|
boolean invert = false;
|
|
|
|
|
for (int i = 0; i < BITMAPS.length; i += 2) {
|
|
|
|
|
int diff = Integer.bitCount(BITMAPS[i] ^ bits);
|
|
|
|
|
if (diff < bestDiff) {
|
|
|
|
|
character = (char) BITMAPS[i + 1];
|
|
|
|
|
bestDiff = diff;
|
|
|
|
|
invert = false;
|
|
|
|
|
}
|
|
|
|
|
diff = Integer.bitCount((~BITMAPS[i]) ^ bits);
|
|
|
|
|
if (diff < bestDiff) {
|
|
|
|
|
character = (char) BITMAPS[i + 1];
|
|
|
|
|
bestDiff = diff;
|
|
|
|
|
invert = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If the match is quite bad, use a shade image instead.
|
2016-04-09 17:15:33 +02:00
|
|
|
|
if (bestDiff > 10) {
|
2016-04-09 00:39:53 +02:00
|
|
|
|
invert = false;
|
|
|
|
|
character = " \u2591\u2592\u2593\u2588".charAt(Math.min(4, fgCount * 5 / 32));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If we use an inverted character, we need to swap the colors.
|
|
|
|
|
if (invert) {
|
|
|
|
|
int[] tmp = bgColor;
|
|
|
|
|
bgColor = fgColor;
|
|
|
|
|
fgColor = tmp;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Roughly modeled after the corresponding HTML 5 class.
|
|
|
|
|
*/
|
|
|
|
|
static class ImageData {
|
|
|
|
|
public final int width;
|
|
|
|
|
public final int height;
|
|
|
|
|
public final byte[] data;
|
|
|
|
|
|
|
|
|
|
public ImageData(int width, int height) {
|
|
|
|
|
this.width = width;
|
|
|
|
|
this.height = height;
|
|
|
|
|
this.data = new byte[width * height * 4];
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-17 03:40:53 +02:00
|
|
|
|
public String hex6(int r, int g, int b) {
|
|
|
|
|
return Integer.toHexString((1 << 24) | ((r & 255) << 16) | ((g & 255) << 8) | (b & 255)).substring(1);
|
|
|
|
|
}
|
2016-04-09 00:39:53 +02:00
|
|
|
|
/**
|
|
|
|
|
* Convert the image to an Ansi control character string setting the colors
|
|
|
|
|
*/
|
2016-06-26 13:09:25 +02:00
|
|
|
|
public String dump(int mode) {
|
2016-04-09 00:39:53 +02:00
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
BlockChar blockChar = new BlockChar();
|
|
|
|
|
|
|
|
|
|
for (int y = 0; y < height - 7; y += 8) {
|
|
|
|
|
int pos = y * width * 4;
|
2016-07-17 03:40:53 +02:00
|
|
|
|
if (html) {
|
|
|
|
|
String last = "";
|
|
|
|
|
for (int x = 0; x < width - 3; x += 4) {
|
|
|
|
|
blockChar.load(data, pos, width * 4);
|
|
|
|
|
String fg = hex6(blockChar.fgColor[0], blockChar.fgColor[1], blockChar.fgColor[2]);
|
|
|
|
|
String bg = hex6(blockChar.bgColor[0], blockChar.bgColor[1], blockChar.bgColor[2]);
|
|
|
|
|
String style = "background-color:#" + bg + ";color:#" + fg;
|
|
|
|
|
if (!style.equals(last)) {
|
|
|
|
|
if (!last.isEmpty()) {
|
|
|
|
|
sb.append("</tt>");
|
|
|
|
|
}
|
|
|
|
|
sb.append("<tt style='").append(style).append("'>");
|
2016-07-17 03:54:48 +02:00
|
|
|
|
last = style;
|
|
|
|
|
}
|
|
|
|
|
sb.append("&#" + ((int) blockChar.character) + ";");
|
2016-07-17 03:40:53 +02:00
|
|
|
|
pos += 16;
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
2016-07-17 03:54:48 +02:00
|
|
|
|
sb.append("</tt><br />\n");
|
2016-07-17 03:40:53 +02:00
|
|
|
|
} else {
|
|
|
|
|
String lastFg = "";
|
|
|
|
|
String lastBg = "";
|
|
|
|
|
for (int x = 0; x < width - 3; x += 4) {
|
|
|
|
|
blockChar.load(data, pos, width * 4);
|
|
|
|
|
String fg = Ansi.color(Ansi.FG | mode, blockChar.fgColor[0], blockChar.fgColor[1], blockChar.fgColor[2]);
|
|
|
|
|
String bg = Ansi.color(Ansi.BG | mode, blockChar.bgColor[0], blockChar.bgColor[1], blockChar.bgColor[2]);
|
|
|
|
|
if (!fg.equals(lastFg)) {
|
|
|
|
|
sb.append(fg);
|
|
|
|
|
lastFg = fg;
|
|
|
|
|
}
|
|
|
|
|
if (!bg.equals(lastBg)) {
|
|
|
|
|
sb.append(bg);
|
|
|
|
|
lastBg = bg;
|
|
|
|
|
}
|
|
|
|
|
sb.append(blockChar.character);
|
|
|
|
|
pos += 16;
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
2016-07-17 03:54:48 +02:00
|
|
|
|
sb.append(Ansi.RESET).append("\n");
|
2016-04-09 00:39:53 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return sb.toString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|