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 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976
|
Barcode Writer in Pure Postscript
Terry Burton tez@terryburton.co.uk
Abstract:
This document describes the implementation of the Barcode Writer in Pure
Postscript project, explains by example how to use this to generate your own
barcodes, and provides a simple reference to using the symbologies that it
supports.
Contents
* Introduction
* Code_Commentary
o The_Barcode_Data_Structure
o An_Encoder
o The_Renderer
o Notes_Regarding_Coding_Style
* Resources_and_Examples
o Language_Specific_APIs
o Front_Ends
o Installing_the_Barcode_Generation_Capability_into_a_Printer's_Virtual
Machine
o Hints_for_Generating_Precisely_the_Required_Symbol
o Printing_in_Perl
* Supported_Symbologies
o EAN-13
o EAN-8
o UPC-A
o UPC-E
o EAN-5
o EAN-2
o ISBN
o Code-39
o Code-128_and_UCC/EAN-128
o Rationalized_Codabar
o Interleaved_2_of_5_and_ITF-14
o Code_2_of_5
o Postnet
o Royal_Mail
o MSI
o Plessey
* License
Introduction
Often there is a need to implement routines in several different languages that
output Adobe Postscript for the purpose of printing barcodes. This project
implements the printing of barcodes entirely within level 2 PostScript. This
means that the entire process of converting the input string into the printed
output is performed by the printer or print system, thus avoiding the need to
later reimplement the barcode generation process when your development language
changes.
To make it as easy as possible to incorporate this project into your own
systems, whether they be freely available or proprietary, it is licensed under
the permissive MIT/X-Consortium License given in section 5.
The project homepage is at http://www.terryburton.co.uk/barcodewriter.
This is the main resource for the project providing the latest downloads of
code and documentation, as well as access to the support and development
mailing list.
Acknowledgements
The author wishes to take this opportunity to thank the small, growing
community of developers that have helped to develop, test and document this
project with their suggestions and code. Most especially Michael Landers and
Ross McFarland for freely giving their encoder implementations, and Lawrence
Horwitz for his suggestions and testing.
Code Commentary
This commentary assumes familiarity with the PostScript language1.
The code is split cleanly into two types of procedure:
The encoders
Each of these represents a barcode symbology2, e.g. EAN-13 or Code-128.
It takes a string containing the barcode data and a string containing a
list of options that modify the output of the encoder. It generates a
structured representation of the barcode and its text for the symbology,
including the calculation of check digits where necessary.
The renderer
This takes the output of an encoder and generates a visual representation
of the barcode.
This means that all barcodes can be generated simply in a similar manner:
(78858101497) (includetext height=0.6) upca barcode
(0123456789) (includecheck) interleaved2of5 barcode
The Barcode Data Structure
The following table describes the structured representation of a barcode that
is passed by an encoder to the renderer as a dictionary when the PostScript is
executed.
_____________________________________________________________________________
|Element______________|Key|Value______________________________________________|
|Space bar succession |sbs|String containing the integer widths, in points, of|
|_____________________|___|each_bar_and_space,_starting_with_the_leftmost_bar.|
|Bar height succession|bhs|Array containing the height of each bar in inches, |
|_____________________|___|starting_with_the_leftmost_bar.____________________|
|Bar base succession |bbs|Array containing the offset of the base of each bar|
|_____________________|___|in_inches,_starting_with_the_leftmost_bar._________|
| | |Array of arrays that contain the character, |
|Human readable text |txt|position, height, font and scale factor (font |
| | |size), in points, for each of the visible text |
|_____________________|___|characters.________________________________________|
|Renderer options |opt|String containing the user-defined renderer |
|_____________________|___|options.___________________________________________|
An Encoder
The procedure labelled code2of5 is a simple example of an encoder, which we
will now consider. Its purpose is to accept as input a string containing the
barcode contents and a string containing a list of options, and to process
these in a way that is specific to this encoder, and finally to output an
instance of the dictionary-based data structure described in section 2.1 that
represents the barcode contents in the Code 2 of 5 symbology.
As with all of the encoders, the input string is assumed to be valid for the
corresponding symbology, otherwise the behaviour is undefined.
The variables that we use in this procedure are confined to local scope by
declaring the procedure as follows:
/code2of5 {
0 begin
...
end
} bind def
/code2of5 load 0 1 dict put
We start by immediately reading the contents strings that are passed as
arguments to this procedure by the user. We duplicate the options string
because it is later passed unamended to the renderer.
/options exch def
/renderopts options def
/barcode exch def
We initialise a few default variables. Those variables corresponding to options
that can be enabled with the options argument are initially set to false.
/includetext false def
/textfont /Courier def
/textsize 10 def
/textpos -7 def
/height 1 def
The options string is tokenised with each successive token defining either a
name value pair which we instantiate or a lone variable that we define as true,
allowing us to override the given default variables given above.
options {
token false eq {exit} if dup length string cvs (=) search
true eq {cvlit exch pop exch def} {cvlit true def} ifelse
} loop
Since any user given options create variables that are strings we need to
convert them back to their intended types.
/textfont textfont cvlit def
/textsize textsize cvr def
/textpos textpos cvr def
/height height cvr def
We then create an array of string encodings for each of the available
characters which we then declare in another string. This information can be
derived from careful reading of the relevant specification, although this is
often surprisingly difficult to obtain.
/encs
[ (1111313111) (3111111131) (1131111131) (3131111111)
(1111311131) (3111311111) (1131311111) (1111113131)
(3111113111) (1131113111) (313111) (311131)
] def
/barchars (0123456789) def
We now store the length of the content string and calculate the total number of
bars and spaces in the resulting barcode. We initialise a string of size
dependant on this length into which we will build the space bar succession.
Similarly, we create an array into which we will add the human readable text
information.
/barlen barcode length def
/sbs barlen 10 mul 12 add string def
/txt barlen array def
We now begin to populate the space bar succession by adding the encoding of the
start character to the beginning.
sbs 0 encs 10 get putinterval
We now enter the main loop which iterates over the content string from start to
finish, looking up the encoding for each character, adding this to the space
bar succession.
It is important to understand how the encoding for a given character is
derived. Firstly, given a character, we find its position in the string of all
available characters. We then use this position to index the array of character
encodings to obtain the encoding for the given character, which is added to the
space/bar succession. Likewise, the character is added to the array of human
readable text along with positioning and font information.
0 1 barlen 1 sub {
/i exch def
barcode i 1 getinterval barchars exch search
pop
length /indx exch def
pop pop
/enc encs indx get def
sbs i 10 mul 6 add enc putinterval
txt i [barcode i 1 getinterval i 14 mul 10 add -7
textfont textsize] put
} for
The encoding for the end character is obtained and added to the end of the
space bar succession.
sbs barlen 10 mul 6 add encs 11 get putinterval
Finally we prepare to push a dictionary containing the space bar succession
(and any additional information defined in section 2.1) that will be passed to
the renderer.
/retval 1 dict def
retval (sbs) sbs put
retval (bhs) [sbs length 1 add 2 idiv {height} repeat] put
retval (bbs) [sbs length 1 add 2 idiv {0} repeat] put
includetext {
retval (txt) txt put
} if
retval (opt) renderopts put
retval
The Renderer
The procedure labelled barcode is known as the renderer, which we now consider.
Its purpose is to accept as input an instance of the dictionary-based data
structure described in section 2.1 that represents a barcode in some arbitrary
symbology and produce a visual rendering of this at the current point.
The variables that we use in this procedure are confined to local scope by
declaring the procedure as follows:
/barcode {
0 begin
...
end
} bind def
/barcode load 0 1 dict put
We then immediately read the dictionary-based data structure which is passed as
a single argument to this procedure by an encoder, from which we extract the
space bar succession, bar height succession and bar base succession.
/args exch def
/sbs args (sbs) get def
/bhs args (bhs) get def
/bbs args (bbs) get def
/renderopts args (opt) get def
We attempt to extract from the dictionary the array containing the information
about human readable text. However, this may not exist in the dictionary in
which case we create a default empty array.
args (txt) known {
/txt args (txt) get def
} {
/txt [] def
} ifelse
Just as with the encoders, we read and tokenise the supplied options allowing
specific rendering options to be overridden.
/inkspread 0.15 def
renderopts {
token false eq {exit} if dup length string cvs (=) search
true eq {cvlit exch pop exch def} {cvlit true def} ifelse
} loop
/inkspread inkspread cvr def
We have extracted or derived all of the necessary information from the input,
and now use the space bar succession, bar height succession and bar base
succession in calculations that create a single array containing elements that
give coordinates for each of the bars in the barcode.
We start by creating a bars array that is half the length of the space bar
succession. We build this by repeatedly adding array elements that contain the
height, x-coordinate, y-coordinate and width of single bars. The height and y-
coordinates are read from the bar height succession and the bar base
succession, respectively, whilst the x-coordinate and the width are made from a
calculation of the total indent, based on the space bar succession and a
compensating factor that accounts for ink spread.
/bars sbs length 1 add 2 idiv array def
/x 0.00 def
0 1 sbs length 1 sub {
/i exch def
/d sbs i get 48 sub def
i 2 mod 0 eq {
/h bhs i 2 idiv get 72 mul def
/c d 2 div x add def
/y bbs i 2 idiv get 72 mul def
/w d inkspread sub def
bars i 2 idiv [h c y w] put
} if
/x x d add def
} for
Finally, we perform the actual rendering in two phases. Firstly we use the
contents of the bars array that we just built to render each of the bars, and
secondly we use the contents of the text array extracted from the input
argument to render the text. We make an efficiency saving here by not
performing loading and rescaling of a font if the scale factor for the font
size is 0. The graphics state is preserved across calls to this procedure to
prevent unexpected interference with the users environment.
gsave
bars {
{} forall
setlinewidth moveto 0 exch rlineto stroke
} forall
txt {
{} forall
dup 0 ne {exch findfont exch scalefont setfont}
{pop pop}
ifelse
moveto show
} forall
grestore
Notes Regarding Coding Style
PostScript programming veterans are encouraged to remember that the majority of
people who read the code are likely to have little, if any, prior knowledge of
the language.
To encourage development, the code has been written with these goals in mind:
-
That it be easy to use and to comprehend
-
That it be easy to modify and enhance
To this end the following points should be observed for all new code
submissions:
-
New encoders should be based on the code of a similar existing encoder
-
Include comments where these clarify the operations involved, particular
where something unexpected happens
-
Prefer simplicity to efficency and clarity to obfuscation, except where
this will be a problem
Resources and Examples
There are several ways of using the PostScript within your own projects.
Many example uses of the code for various languages and platforms can be
downloaded from the code repository at http://www.terryburton.co.uk/
barcodewriter/files/repository/
Language Specific APIs
No language specific APIs exist yet. If you have experience writing API
specifications and would like to help create an API design for the project then
contact the author.
Front Ends
The following is a list of the front ends available for the project.
Web based demo
http://www.raise-the-bar.co.uk/demo
pst-barcode
pst-barcode is a PSTricks package for LATEX.
http://www.ctan.org/tex-archive/graphics/pstricks/contrib/pst-barcode/
Installing the Barcode Generation Capability into a Printer's Virtual Machine
Most genuine PostScript printers allow procedures to be defined such that they
persist across different jobs through the use of the exitserver command. If
your printer supports this then you will be able to print the main code
containing the definitons of all the encoders and the renderer once, e.g. soon
after the device is turned on, and later omit these definitons from each of the
barcode documents that you print.
To install the barcode generation capabilities into the virtual machine of a
PostScript printer you need to uncomment a line near the top of the code so
that it reads:
serverdict begin 0 exitserver
Once this code is printed the procedural definitions for the encoders and the
renderer will remain defined across all jobs until the device is reset either
by power-cycling or with the following code:
serverdict begin 0 exitserver systemdict /quit get exec
Hints for Generating Precisely the Required Symbol
To create a barcode to a required width and height, without stretching the
human readable text, perform the following steps.
Create a basic symbol by choosing the relevant data and text options for the
corresponding encoder, and position this using translate such that the bottom-
left corner of the bars is in the required location:
gsave
430 750 translate
(977147396801) (includetext) ean13 barcode
grestore
Find the uniform scale factor (same value for x and y) that makes your output
of the required width:
gsave
430 750 translate
1.3 1.3 scale % <-- Add a line like this
(977147396801) (includetext) ean13 barcode
grestore
Add a height option that adjusts the bar height appropriately (taking the
scaling into account):
gsave
430 750 translate
1.3 1.3 scale
% Added height=0.8 option to adjust height
(977147396801) (includetext height=0.8) ean13 barcode
grestore
The result should now be of the intended dimensions at the desired location
with properly scaled text. You can now add any additional options to customise
the symbol.
Printing in Perl
This example will print a page of EAN-13s ranging between two given values when
called from a shell like this:
$ ./ean13s.pl 978186074271 978186074292 | lpr
The contents of the script ean13s.pl is as follows:
#!/usr/bin/perl -w
use strict;
die 'Requires two arguments' if (@ARGV!=2);
open(PS,'barcode.ps') || die 'File not found';
$_=join('',<PS>);
close(PS);
print "%!PS-Adobe-2.0\n";
m/
%\ --BEGIN\ TEMPLATE--
(.*)
%\ --END\ TEMPLATE--
/sx || die 'Unable to parse out the template';
print $1;
for (my $i=$ARGV[0], my $j=0; $i<$ARGV[1]; $i++, $j++) {
my $x=100+150*(int($j/7));
my $y=100+100*($j%7);
print "gsave\n";
print "$x $y translate\n";
print "($i) (includetext) ean13 barcode\n";
print "grestore\n";
}
print "showpage\n";
Supported Symbologies
The following section shows the symbologies that are supported by the encoders,
including the available features for each. This list may not be up-to-date. If
it does not contain any of the formats or features that you require then check
the project source code or try the support mailing list.
EAN-13
Data
12 or 13 digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Notes
If just 12 digits are entered then the check digit is calculated
automatically
Figure
1:
(9781860742712)
(includetext
guardwhitespace)
ean13
barcode
EAN-8
Data
8 digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Figure
2:
(12345678)
(includetext
guardwhitespace
height=0.6)
ean8
barcode
UPC-A
Data
11 or 12 digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Notes
If just 11 digits are entered then the check digit is calculated
automatically
Figure
3:
(78858101497)
(includetext)
upca
barcode
UPC-E
Data
7 or 8 digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Notes
If just 7 digits are entered then the check digit is calculated
automatically
Figure
4:
(0123456)
(includetext)
upce
barcode
EAN-5
Data
5 digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Figure
5:
(90200)
(includetext
guardwhitespace)
ean5
barcode
EAN-2
Data
2 digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Figure
6:
(38)
(includetext
guardwhitespace)
ean2
barcode
ISBN
Data
9 or 10 digits seperated appropriately with dashes
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Notes
If just 9 digits are entered then the human readable ISBN check digit is
calculated automatically
Figure
7:
(1-
58880-
149)
(includetext)
isbn
barcode
Code-39
Data
Variable number of characters, digits and any of the symbols -. *$/+%.
Options
___________________________________________________
|Option____________|Feature_________________________|
|includecheck______|Enable_check_digit______________|
|includetext_______|Enable_human_readable_text______|
|includecheckintext|Make_check_digit_visible_in_text|
Figure
8:
(CODE-
39)
(includecheck
includetext)
code39
barcode
Code-128 and UCC/EAN-128
Data
Variable number of ASCII characters and special funtion symbols, starting
with the approriate start character for the initial character set. UCC/
EAN-128s must have a manditory FNC 1 symbol immediately following the
start character.
Options
___________________________________________________
|Option____________|Feature_________________________|
|includetext_______|Enable_human_readable_text______|
|includecheckintext|Make_check_digit_visible_in_text|
Notes
Any non-printable character can be entered via its escaped ordinal value,
for example ^070 for ACK and ^102 for FNC 1. Since a caret symbol serves
as an escape character it must be escaped as ^062 if used in the data.
The check character is always added automatically.
Figure
9:
(^104^102Count^0991234^101!)
(includetext)
code128
barcode
Rationalized Codabar
Data
Variable number of digits and any of the symbols -$:/.+ABCD.
Options
___________________________________________________
|Option____________|Feature_________________________|
|includecheck______|Enable_check_digit______________|
|includetext_______|Enable_human_readable_text______|
|includecheckintext|Make_check_digit_visible_in_text|
Figure
10:
(0123456789)
(includetext)
rationalizedCodabar
barcode
Interleaved 2 of 5 and ITF-14
Data
Variable number of digits. An ITF-14 is 14 characters and does not have a
check digit.
Options
___________________________________________________
|Option____________|Feature_________________________|
|includecheck______|Enable_check_digit______________|
|includetext_______|Enable_human_readable_text______|
|includecheckintext|Make_check_digit_visible_in_text|
Notes
The data may be automatically prefixed with 0 to make the data, including
optional check digit, of even length.
Figure
11:
(05012345678900)
(includecheck
height=0.7)
interleaved2of5
barcode
Code 2 of 5
Data
Variable number of digits
Options
______________________________________
|Option_____|Feature___________________|
|includetext|Enable_human_readable_text|
Figure
12:
(0123456789)
(includetext
textpos=75
textfont=Helvetica
textsize=16)
code2of5
barcode
Postnet
Data
Variable number digits
Options
___________________________________________________________
|Option____________|Feature_________________________________|
|includetext_______|Enable_human_readable_text______________|
|includecheckintext|Make_the_check_digit_visible_in_the_text|
Notes
Check digit is always added automatically
Figure
13:
(01234567)
(includetext
textpos=-
10
textfont=Arial
textsize=10)
postnet
barcode
Royal Mail
Data
Variable number digits and capital letters
Options
___________________________________________________________
|Option____________|Feature_________________________________|
|includetext_______|Enable_human_readable_text______________|
|includecheckintext|Make_the_check_digit_visible_in_the_text|
Notes
Check digit is always added automatically
Figure
14:
(LE28HS9Z)
(includetext)
royalmail
barcode
MSI
Data
Variable number digits
Options
_______________________________________________________
|Option____________|Feature_____________________________|
|includecheck______|Enable_check_digit__________________|
|includetext_______|Enable_human_readable_text__________|
|includecheckintext|Make_check_digit_visible_in_the_text|
Figure
15:
(0123456789)
(includecheck
includetext)
msi
barcode
Plessey
Data
Variable number of hexadecimal characters
Options
____________________________________________________________
|Option____________|Feature__________________________________|
|includetext_______|Enable_human_readable_text_______________|
|includecheckintext|Make_the_check_digits_visible_in_the_text|
Notes
Check digits are always added automatically.
Figure
16:
(012345ABCDEF)
(includetext)
plessey
barcode
License
Copyright 2004 Terry Burton
Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
the Software without restriction, including without limitation the rights to
use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
of the Software, and to permit persons to whom the Software is furnished to do
so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
The software is provided "as is", without warranty of any kind, express or
implied, including but not limited to the warranties of merchantability,
fitness for a particular purpose and noninfringement. In no event shall the
authors or copyright holders be liable for any claim, damages or other
liability, whether in an action of contract, tort or otherwise, arising from,
out of or in connection with the software or the use or other dealings in the
software.
-------------------------------------------------------------------------------
Footnotes
... language1
The PostScript Language Tutorial and Cookbook (a.k.a. the Blue Book),
which is freely available online, serves as both a useful tutorial and
reference manual to the language.
... symbology2
By symbology we mean an accepted standard for representation of data as a
barcode
-------------------------------------------------------------------------------
http://www.terryburton.co.uk/barcodewriter
|