forked from dpavlin/Printer-Zebra
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpbm2ZPL.pl
executable file
·129 lines (107 loc) · 3.32 KB
/
pbm2ZPL.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
#!/usr/bin/perl
use warnings;
use strict;
use autodie;
use Data::Dump qw(dump);
# DG compression is documented in ZPL II Programming Guide Volume Two, page 71-72
my $darkness = $ENV{DARKNESS} || 0;
my $compress = $ENV{COMPRESS} || 1;
my $pnm_file = shift @ARGV || die "usage: $0 print.pnm > print.zpl\n";
open(my $fh, '<', $pnm_file);
my $magic = <$fh>; chomp $magic;
my $size = <$fh>;
while ( $size =~ m/^#/ ) { $size = <$fh> }; # skip comments
chomp $size;
my ( $w, $h ) = split(/ /,$size,2);
warn "WARNING: width of $pnm_file not 832 but $w !\n" if $w != 832;
my $bitmap;
if ( $magic eq 'P4' ) {
local $/ = undef;
$bitmap = <$fh>;
} elsif ( $magic eq 'P6' ) {
my $max_color = <$fh>; chomp $max_color;
my $trashold = $max_color / 2;
local $/ = undef;
my $rgb = <$fh>;
my $mask = 0x80;
my $byte = 0;
my $o = 0;
while ( $o < length($rgb) ) {
my $px = ord(substr($rgb,$o,1)); $o += 3;
$byte ^= $mask if $px < $trashold;
$mask >>= 1;
if ( ! $mask ) {
$bitmap .= chr($byte);
$byte = 0;
$mask = 0x80;
}
}
warn dump $bitmap;
} else {
die "$pnm_file magick $magic not supported\n";
}
print '^XA';
printf '~TA%03d', 0; # tear-off
print '~JSN'; # sensor detect N = normal, 90%
#print '^LT18'; # label top -120 .. 120
print '^MNW'; # media tracking N = continuous Y/W = web sensing M = mark sensing
print '^MTD'; # media type T = termal D = direct (ribbon!)
print '^PON'; # print orientation N = normal I = invert
print '^PMN'; # print mirror Y/N
print '^LH0,0'; # label home x,y
print '^JMA'; # dots/mm A = 24/12/8/6 B = 12/6/4/3
print '^PR4,4'; # print,slew,backfeed speed in inch/s 2 .. 12 [default: 2,6,2]
printf '^MD%d', $darkness ; # media darkness -30 .. 30 / XiIIIPlus 0..30/0.1 increments
print '^JUS'; # configuration update F = factory default R = recall S = save
print '^LRN'; # label reverse Y/N
print '^CI0'; # change international font 0..255
print "^XZ\r\n";
printf "~DG000.GRF,%d,%d,\r\n", $w / 8 * $h, $w / 8;
my $last_line = '';
sub zpl_compress {
my $compress = shift;
my $repeat = length($compress);
my $out;
while ( $repeat >= 400 ) {
$out .= 'z';
$repeat -= 400;
}
if ( $repeat >= 20 ) {
$out .= chr( ord('f') + ( $repeat / 20 ) );
$repeat %= 20;
}
if ( $repeat > 0 ) {
$out .= chr( ord('F') + $repeat );
}
$out .= substr($compress,0,1); # char
warn "## zpl_compress $repeat = $compress -> $out\n";
return $out;
}
foreach my $y ( 0 .. $h - 1 ) {
my $line = substr( $bitmap, $y * ( $w / 8 ), $w / 8 );
if ( $line eq $last_line ) {
print ':';
warn "# $y repeat previous line\n";
} else {
my $hex = unpack('H*', $line);
if ( $compress ) {
$last_line = $line;
$hex =~ s/0+$/,/ && warn "# $y fill 0 to right\n";
$hex =~ s/F+$/!/i && warn "# $y fill 1 to right\n";
$hex =~ s/((.)\2+)/zpl_compress($1)/egs;
}
print $hex;
}
}
print '^XA';
print '^MMT'; # print mode,prepeel T=tear-off P=peel-off R=rewind A=applicator C=cutter, Y/N
printf '^LL%d', $h; # label length FIXME ignore empty bottom
printf '^PW%d', $w; # print width
print '^LS0'; # label shift -9999..9999
printf '^FT%d,%d', 0, $h; # field typeset x,y graphic origin is bottom-left
print '^XG000.GRF,1,1^FS'; # recall grapmic source/name,magnification_x,magnification_y
print '^PQ1,0,1,Y'; # print quantity total,pause/cut,replicates,no_pause
print "^XZ\r\n";
print '^XA';
print '^ID000.GRF^FS'; # object delete
print "^XZ\r\n";