The rendering of the fonts is very difficult in low resolution we are working. The screen is 72 pixels per inch or 96 pixels per inch, this is less than the quality of a fax. Printers are 300 or 600 DPI (dots per inch), photo setting is even 1500 DPI. Adobe uses some hints to display fonts properly with low resolutions, but must problems go away with more resolution.
What we can do here, is oversampling. For each pixel of the canvas, we calculate multiple pixels and then reduce the size of the result.
To enable oversampling, we need to adapt our code. First we create an operator setpagedevice. This operator gives information on the output device in form of a dictionary. It will pass many values to the output device in later chapters, for now it's just the key /oversampling. Note that setpagedevice is a standard PostScript operator but the content of the dictionary is not.
To apply the oversampling, we need to adapt rpnContext, scannFill and postScriptEditor. There is a new device dictonary in rpnContext with the oversampling value with default 1. Scannfill will scale the x, y, width and height values. And postScriptEditor will provide a larger ImageData object and we need also an offscreen Canvas to scale the image (two steps).
This is the text without oversampling.
Run
postScriptEditor(`(CMUSerif-BoldItalic) run
/CMUSerif-BoldItalic findfont 48 scalefont setfont
0 250 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 36 scalefont setfont
0 200 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 24 scalefont setfont
0 160 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 12 scalefont setfont
0 135 moveto (CMUSerif-BoldItalic) show
`);
Run (CMUSerif-BoldItalic) run
/CMUSerif-BoldItalic findfont 48 scalefont setfont
0 250 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 36 scalefont setfont
0 200 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 24 scalefont setfont
0 160 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 12 scalefont setfont
0 135 moveto (CMUSerif-BoldItalic) show
This one is with oversampling 2.
Run
postScriptEditor(`10 dict begin /oversampling 2 def currentdict setpagedevice end
(CMUSerif-BoldItalic) run
/CMUSerif-BoldItalic findfont 48 scalefont setfont
0 250 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 36 scalefont setfont
0 200 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 24 scalefont setfont
0 160 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 12 scalefont setfont
0 135 moveto (CMUSerif-BoldItalic) show
`);
Run 10 dict begin /oversampling 2 def currentdict setpagedevice end
(CMUSerif-BoldItalic) run
/CMUSerif-BoldItalic findfont 48 scalefont setfont
0 250 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 36 scalefont setfont
0 200 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 24 scalefont setfont
0 160 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 12 scalefont setfont
0 135 moveto (CMUSerif-BoldItalic) show
And this one is with oversampling 4.
Run
postScriptEditor(`10 dict begin /oversampling 4 def currentdict setpagedevice end
(CMUSerif-BoldItalic) run
/CMUSerif-BoldItalic findfont 48 scalefont setfont
0 250 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 36 scalefont setfont
0 200 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 24 scalefont setfont
0 160 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 12 scalefont setfont
0 135 moveto (CMUSerif-BoldItalic) show
`);
Run 10 dict begin /oversampling 4 def currentdict setpagedevice end
(CMUSerif-BoldItalic) run
/CMUSerif-BoldItalic findfont 48 scalefont setfont
0 250 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 36 scalefont setfont
0 200 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 24 scalefont setfont
0 160 moveto (CMUSerif-BoldItalic) show
/CMUSerif-BoldItalic findfont 12 scalefont setfont
0 135 moveto (CMUSerif-BoldItalic) show
The device has to be set before any drawing operators. Here is our justified text.
Run
postScriptEditor(`10 dict begin /oversampling 2 def currentdict setpagedevice end
% prologue
% support procedures
% n1 n2 max result
/max { 2 copy gt { pop } { exch pop } ifelse } def
% text countblanks n
/countblanks { 10 dict begin /c 0 def { ( ) search { /c c 1 add def pop pop } { pop c end exit } ifelse } loop } def
% text width justifyshow -
/justifyshow { exch dup stringwidth pop 3 -1 roll exch sub exch dup countblanks 1 max 3 -1 roll exch div 0 32 4 -1 roll widthshow } def
% text width wraptext post width pre
/wraptext { 10 dict begin /maxwidth exch def /text exch def
/linewidth 0 def
/linelength 0 def
/spaces 0 def
/spacewidth 0 0 moveto ( ) stringwidth pop def
text
{
% search leaves the post on the stack, so we assign it to currenttext
dup /currenttext exch def
( ) search {
% add word width and check if linewidth is still not bigger than max
dup stringwidth pop linewidth add dup spaces add maxwidth le
% add space and count characters
{ /linewidth exch def
length 1 add linelength add /linelength exch def
/spaces spaces spacewidth add def
pop }
% check if half space ok
{
spaces 2 div add maxwidth le
{ length 1 add linelength add /linelength exch def exit}
{ pop pop pop exit }
ifelse
} ifelse
}
% no space found we are at the last word
{ pop text length /linelength exch def /currenttext () def exit } ifelse } loop
% remove trailing space character if not empty text
currenttext length { /linelength linelength 1 sub def } if
% set text if only one word
linelength 0 le { /linelength currenttext length def } if
% output
currenttext maxwidth text 0 linelength getinterval end
} def
% text x y w l showjustifyblock y
/showjustifyblock {
/l exch def
/w exch def
/y exch def
/x exch def w
{ wraptext 3 -1 roll dup 4 1 roll length { newpath x y moveto w justifyshow y l sub /y exch def } { x y moveto show pop pop exit } ifelse } loop y } def
% parameter
/bodystart 300 def
/linespace 33 def
/LM 20 def
/RM 570 def
(CMUSerif-Roman) run
/CMUSerif-Roman findfont 33 scalefont setfont
% main
/linewidth RM LM sub def
/nextline bodystart def
(Acme Widgets was founded in 1952 by Dippy and Daffy Acme to produce high technology widgets for the growing aerospace market. Acme was quickly recognized as being the best widget works in the country. Continued investment in new technology and manufacturing methods has kept Acme Widgets in the forefront of this industry.)
LM nextline linewidth linespace showjustifyblock pop
showpage
`);
Run 10 dict begin /oversampling 2 def currentdict setpagedevice end
% prologue
% support procedures
% n1 n2 max result
/max { 2 copy gt { pop } { exch pop } ifelse } def
% text countblanks n
/countblanks { 10 dict begin /c 0 def { ( ) search { /c c 1 add def pop pop } { pop c end exit } ifelse } loop } def
% text width justifyshow -
/justifyshow { exch dup stringwidth pop 3 -1 roll exch sub exch dup countblanks 1 max 3 -1 roll exch div 0 32 4 -1 roll widthshow } def
% text width wraptext post width pre
/wraptext { 10 dict begin /maxwidth exch def /text exch def
/linewidth 0 def
/linelength 0 def
/spaces 0 def
/spacewidth 0 0 moveto ( ) stringwidth pop def
text
{
% search leaves the post on the stack, so we assign it to currenttext
dup /currenttext exch def
( ) search {
% add word width and check if linewidth is still not bigger than max
dup stringwidth pop linewidth add dup spaces add maxwidth le
% add space and count characters
{ /linewidth exch def
length 1 add linelength add /linelength exch def
/spaces spaces spacewidth add def
pop }
% check if half space ok
{
spaces 2 div add maxwidth le
{ length 1 add linelength add /linelength exch def exit}
{ pop pop pop exit }
ifelse
} ifelse
}
% no space found we are at the last word
{ pop text length /linelength exch def /currenttext () def exit } ifelse } loop
% remove trailing space character if not empty text
currenttext length { /linelength linelength 1 sub def } if
% set text if only one word
linelength 0 le { /linelength currenttext length def } if
% output
currenttext maxwidth text 0 linelength getinterval end
} def
% text x y w l showjustifyblock y
/showjustifyblock {
/l exch def
/w exch def
/y exch def
/x exch def w
{ wraptext 3 -1 roll dup 4 1 roll length { newpath x y moveto w justifyshow y l sub /y exch def } { x y moveto show pop pop exit } ifelse } loop y } def
% parameter
/bodystart 300 def
/linespace 33 def
/LM 20 def
/RM 570 def
(CMUSerif-Roman) run
/CMUSerif-Roman findfont 33 scalefont setfont
% main
/linewidth RM LM sub def
/nextline bodystart def
(Acme Widgets was founded in 1952 by Dippy and Daffy Acme to produce high technology widgets for the growing aerospace market. Acme was quickly recognized as being the best widget works in the country. Continued investment in new technology and manufacturing methods has kept Acme Widgets in the forefront of this industry.)
LM nextline linewidth linespace showjustifyblock pop
showpage
We have now 2115 lines ps20240829.js
My Journey to PostScript