Support Full-ASCII in Code93Writer (#1104)

* support extended characters in Code93Writer

* remove redundancy

* add test for convertToExtended

* thanks for review

* fix comment, use package-private for testing
This commit is contained in:
MakKi (makki_d) 2018-11-09 23:05:07 +09:00 committed by Sean Owen
parent b31e090239
commit e636498b5e
3 changed files with 109 additions and 27 deletions

View file

@ -52,7 +52,7 @@ public final class Code93Reader extends OneDReader {
0x12E, 0x1D4, 0x1D2, 0x1CA, 0x16E, 0x176, 0x1AE, // - - % 0x12E, 0x1D4, 0x1D2, 0x1CA, 0x16E, 0x176, 0x1AE, // - - %
0x126, 0x1DA, 0x1D6, 0x132, 0x15E, // Control chars? $-* 0x126, 0x1DA, 0x1D6, 0x132, 0x15E, // Control chars? $-*
}; };
private static final int ASTERISK_ENCODING = CHARACTER_ENCODINGS[47]; static final int ASTERISK_ENCODING = CHARACTER_ENCODINGS[47];
private final StringBuilder decodeRowResult; private final StringBuilder decodeRowResult;
private final int[] counters; private final int[] counters;

View file

@ -38,46 +38,44 @@ public class Code93Writer extends OneDimensionalCodeWriter {
return super.encode(contents, format, width, height, hints); return super.encode(contents, format, width, height, hints);
} }
/**
* @param contents barcode contents to encode. It should not be encoded for extended characters.
* @return a {@code boolean[]} of horizontal pixels (false = white, true = black)
*/
@Override @Override
public boolean[] encode(String contents) { public boolean[] encode(String contents) {
contents = convertToExtended(contents);
int length = contents.length(); int length = contents.length();
if (length > 80) { if (length > 80) {
throw new IllegalArgumentException( throw new IllegalArgumentException(
"Requested contents should be less than 80 digits long, but got " + length); "Requested contents should be less than 80 digits long after converting to extended encoding, but got " + length);
} }
//each character is encoded by 9 of 0/1's
int[] widths = new int[9];
//length of code + 2 start/stop characters + 2 checksums, each of 9 bits, plus a termination bar //length of code + 2 start/stop characters + 2 checksums, each of 9 bits, plus a termination bar
int codeWidth = (contents.length() + 2 + 2) * 9 + 1; int codeWidth = (contents.length() + 2 + 2) * 9 + 1;
//start character (*)
toIntArray(Code93Reader.CHARACTER_ENCODINGS[47], widths);
boolean[] result = new boolean[codeWidth]; boolean[] result = new boolean[codeWidth];
int pos = appendPattern(result, 0, widths);
//start character (*)
int pos = appendPattern(result, 0, Code93Reader.ASTERISK_ENCODING);
for (int i = 0; i < length; i++) { for (int i = 0; i < length; i++) {
int indexInString = Code93Reader.ALPHABET_STRING.indexOf(contents.charAt(i)); int indexInString = Code93Reader.ALPHABET_STRING.indexOf(contents.charAt(i));
toIntArray(Code93Reader.CHARACTER_ENCODINGS[indexInString], widths); pos += appendPattern(result, pos, Code93Reader.CHARACTER_ENCODINGS[indexInString]);
pos += appendPattern(result, pos, widths);
} }
//add two checksums //add two checksums
int check1 = computeChecksumIndex(contents, 20); int check1 = computeChecksumIndex(contents, 20);
toIntArray(Code93Reader.CHARACTER_ENCODINGS[check1], widths); pos += appendPattern(result, pos, Code93Reader.CHARACTER_ENCODINGS[check1]);
pos += appendPattern(result, pos, widths);
//append the contents to reflect the first checksum added //append the contents to reflect the first checksum added
contents += Code93Reader.ALPHABET_STRING.charAt(check1); contents += Code93Reader.ALPHABET_STRING.charAt(check1);
int check2 = computeChecksumIndex(contents, 15); int check2 = computeChecksumIndex(contents, 15);
toIntArray(Code93Reader.CHARACTER_ENCODINGS[check2], widths); pos += appendPattern(result, pos, Code93Reader.CHARACTER_ENCODINGS[check2]);
pos += appendPattern(result, pos, widths);
//end character (*) //end character (*)
toIntArray(Code93Reader.CHARACTER_ENCODINGS[47], widths); pos += appendPattern(result, pos, Code93Reader.ASTERISK_ENCODING);
pos += appendPattern(result, pos, widths);
//termination bar (single black bar) //termination bar (single black bar)
result[pos] = true; result[pos] = true;
@ -85,13 +83,6 @@ public class Code93Writer extends OneDimensionalCodeWriter {
return result; return result;
} }
private static void toIntArray(int a, int[] toReturn) {
for (int i = 0; i < 9; i++) {
int temp = a & (1 << (8 - i));
toReturn[i] = temp == 0 ? 0 : 1;
}
}
/** /**
* @param target output to append to * @param target output to append to
* @param pos start position * @param pos start position
@ -102,16 +93,20 @@ public class Code93Writer extends OneDimensionalCodeWriter {
*/ */
@Deprecated @Deprecated
protected static int appendPattern(boolean[] target, int pos, int[] pattern, boolean startColor) { protected static int appendPattern(boolean[] target, int pos, int[] pattern, boolean startColor) {
return appendPattern(target, pos, pattern);
}
private static int appendPattern(boolean[] target, int pos, int[] pattern) {
for (int bit : pattern) { for (int bit : pattern) {
target[pos++] = bit != 0; target[pos++] = bit != 0;
} }
return 9; return 9;
} }
private static int appendPattern(boolean[] target, int pos, int a) {
for (int i = 0; i < 9; i++) {
int temp = a & (1 << (8 - i));
target[pos + i] = temp != 0;
}
return 9;
}
private static int computeChecksumIndex(String contents, int maxWeight) { private static int computeChecksumIndex(String contents, int maxWeight) {
int weight = 1; int weight = 1;
int total = 0; int total = 0;
@ -125,4 +120,67 @@ public class Code93Writer extends OneDimensionalCodeWriter {
} }
return total % 47; return total % 47;
} }
static String convertToExtended(String contents) {
int length = contents.length();
StringBuilder extendedContent = new StringBuilder(length * 2);
for (int i = 0; i < length; i++) {
char character = contents.charAt(i);
// ($)=a, (%)=b, (/)=c, (+)=d. see Code93Reader.ALPHABET_STRING
if (character == 0) {
// NUL: (%)U
extendedContent.append("bU");
} else if (character <= 26) {
// SOH - SUB: ($)A - ($)Z
extendedContent.append('a');
extendedContent.append((char) ('A' + character - 1));
} else if (character <= 31) {
// ESC - US: (%)A - (%)E
extendedContent.append('b');
extendedContent.append((char) ('A' + character - 27));
} else if (character == ' ' || character == '$' || character == '%' || character == '+') {
// space $ % +
extendedContent.append(character);
} else if (character <= ',') {
// ! " # & ' ( ) * ,: (/)A - (/)L
extendedContent.append('c');
extendedContent.append((char) ('A' + character - '!'));
} else if (character <= '9') {
extendedContent.append(character);
} else if (character == ':') {
// :: (/)Z
extendedContent.append("cZ");
} else if (character <= '?') {
// ; - ?: (%)F - (%)J
extendedContent.append('b');
extendedContent.append((char) ('F' + character - ';'));
} else if (character == '@') {
// @: (%)V
extendedContent.append("bV");
} else if (character <= 'Z') {
// A - Z
extendedContent.append(character);
} else if (character <= '_') {
// [ - _: (%)K - (%)O
extendedContent.append('b');
extendedContent.append((char) ('K' + character - '['));
} else if (character == '`') {
// `: (%)W
extendedContent.append("bW");
} else if (character <= 'z') {
// a - z: (*)A - (*)Z
extendedContent.append('d');
extendedContent.append((char) ('A' + character - 'a'));
} else if (character <= 127) {
// { - DEL: (%)P - (%)T
extendedContent.append('b');
extendedContent.append((char) ('P' + character - '{'));
} else {
throw new IllegalArgumentException(
"Requested content contains a non-encodable character: '" + character + "'");
}
}
return extendedContent.toString();
}
} }

View file

@ -36,6 +36,22 @@ public final class Code93WriterTestCase extends Assert {
"101100101101011001101001101100101101100110101011011001011001101001101101001110101000" + "101100101101011001101001101100101101100110101011011001011001101001101101001110101000" +
"101001010010001010001001010000101001010001001001001001000101010100001000100101000010" + "101001010010001010001001010000101001010001001001001001000101010100001000100101000010" +
"10100111010101000010101011110100000"); "10100111010101000010101011110100000");
doTest("\u0000\u0001\u001a\u001b\u001f $%+!,09:;@AZ[_`az{\u007f",
"00000" + "101011110" +
"111011010" + "110010110" + "100100110" + "110101000" + // bU aA
"100100110" + "100111010" + "111011010" + "110101000" + // aZ bA
"111011010" + "110010010" + "111010010" + "111001010" + // bE space $
"110101110" + "101110110" + "111010110" + "110101000" + // % + cA
"111010110" + "101011000" + "100010100" + "100001010" + // cL 0 9
"111010110" + "100111010" + "111011010" + "110001010" + // cZ bF
"111011010" + "110011010" + "110101000" + "100111010" + // bV A Z
"111011010" + "100011010" + "111011010" + "100101100" + // bK bO
"111011010" + "101101100" + "100110010" + "110101000" + // bW dA
"100110010" + "100111010" + "111011010" + "100010110" + // dZ bP
"111011010" + "110100110" + // bT
"110100010" + "110101100" + // checksum: 12 28
"101011110" + "100000");
} }
private static void doTest(String input, CharSequence expected) throws WriterException { private static void doTest(String input, CharSequence expected) throws WriterException {
@ -43,4 +59,12 @@ public final class Code93WriterTestCase extends Assert {
assertEquals(expected, BitMatrixTestCase.matrixToString(result)); assertEquals(expected, BitMatrixTestCase.matrixToString(result));
} }
@Test
public void testConvertToExtended() throws Exception {
// non-extended chars are not changed.
String src = "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ-. $/+%";
String dst = Code93Writer.convertToExtended(src);
assertEquals(src, dst);
}
} }