moved from ../pdf2swf/font2swf.cc, rewritten.
[swftools.git] / pdf2swf / font2swf.cc
1 /* makefonts.cc
2
3    Utility for converting Type 1 fonts to SWF.
4    
5    Part of the swftools package.
6
7    Copyright (c) 2001 Matthias Kramm <kramm@quiss.org>
8  
9    This program is free software; you can redistribute it and/or modify
10    it under the terms of the GNU General Public License as published by
11    the Free Software Foundation; either version 2 of the License, or
12    (at your option) any later version.
13
14    This program is distributed in the hope that it will be useful,
15    but WITHOUT ANY WARRANTY; without even the implied warranty of
16    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17    GNU General Public License for more details.
18
19    You should have received a copy of the GNU General Public License
20    along with this program; if not, write to the Free Software
21    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */
22
23 #include <stdio.h>
24 #include <unistd.h>
25 #include <fcntl.h>
26 #include <math.h>
27 #include "../lib/rfxswf.h"
28 #include "../lib/args.h"
29 #include "swfoutput.h"
30 #include "spline.h"
31
32 static char * filenames[256];
33 static int filenum;
34 static char * destfilename = "output.swf";
35 int all=0;
36 int verbose=0;
37
38 struct options_t options[] =
39 {
40  {"o","output"},
41  {"v","verbose"},
42  {"V","version"},
43  {0,0}
44 };
45 int args_callback_option(char*name,char*val)
46 {
47     if(!strcmp(name, "V")) {
48         printf("font2swf - part of %s %s\n", PACKAGE, VERSION);
49         exit(0);
50     }
51     else if(!strcmp(name, "o")) {
52         destfilename = val;
53         return 1;
54     }
55     else if(!strcmp(name, "v")) {
56         verbose ++;
57         return 0;
58     }
59     else {
60         printf("Unknown option: -%s\n", name);
61         exit(1);
62     }
63     return 0;
64 }
65 int args_callback_longoption(char*name,char*val)
66 {
67     return args_long2shortoption(options, name, val);
68 }
69 void args_callback_usage(char*name)
70 {    
71     printf("Usage: %s file.afm [...]\n", name);
72     printf("   OR: %s all\n", name);
73     printf("\n");
74     printf("\tIf \"all\" is given instead of font names, all standard fonts\n");
75     printf("\t(Courier, Arial etc.) will be created\n");
76     printf("\n");
77     printf("Options:\n");
78     printf("\n");
79     printf("\t-h , --help\t\t Print help and exit\n");
80     printf("\t-o , --output filename\t set output filename\n");
81     printf("\t-v , --verbose\t\t Be more verbose\n");
82     printf("\t-V , --version\t\t Print program version and exit\n");
83 }
84 int args_callback_command(char*name,char*val)
85 {
86     if(!strcmp(name, "all"))
87         all = 1;
88     else {
89         filenames[filenum++] = strdup(name);
90     }
91     return 0;
92 }
93
94
95 #define standardEncodingSize 256
96 #define symbolEncodingSize 256
97 #define zapfDingbatsEncodingSize 256
98 #define macRomanEncodingSize 256
99 #define winAnsiEncodingSize 256
100
101 extern char *winAnsiEncoding[winAnsiEncodingSize];
102 extern char *standardEncoding[standardEncodingSize];
103 extern char *symbolEncoding[symbolEncodingSize];
104 extern char *zapfDingbatsEncoding[zapfDingbatsEncodingSize];
105 extern char *macRomanEncoding[macRomanEncodingSize];
106
107 extern void drawpath(TAG*tag, T1_OUTLINE*outline, struct swfmatrix*m, int log);
108 extern void resetdrawer();
109 extern void moveto(TAG*tag, plotxy p0);
110 extern void lineto(TAG*tag, plotxy p0);
111
112 SWFFONT * t1font2swffont(int i)
113 {
114     T1_LoadFont(i);
115
116     float angle = T1_GetItalicAngle(i);
117     char*fontname = T1_GetFontName(i);
118     char*fullname = T1_GetFullName(i);
119     char*familyname = T1_GetFamilyName(i);
120     float underline = T1_GetUnderlinePosition(i);
121     BBox bbox = T1_GetFontBBox(i);
122
123     /* if "all" is given, translate the font names in something more
124        readable */
125     if(!strcmp(fullname, "Nimbus Roman No9 L Regular")) fontname = "Helvetica";
126     if(!strcmp(fullname, "Nimbus Roman No9 L Regular Italic")) fontname = "HelveticaItalic";
127     if(!strcmp(fullname, "Nimbus Roman No9 L Medium")) fontname = "HelveticaBold";
128     if(!strcmp(fullname, "Nimbus Roman No9 L Medium Italic")) fontname = "HelveticaBoldItalic";
129     if(!strcmp(fullname, "Nimbus Sans L Regular")) fontname = "Times";
130     if(!strcmp(fullname, "Nimbus Sans L Regular Italic")) fontname = "TimesItalic";
131     if(!strcmp(fullname, "Nimbus Sans L Bold")) fontname = "TimesBold";
132     if(!strcmp(fullname, "Nimbus Sans L Bold Italic")) fontname = "TimesBoldItalic";
133     if(!strcmp(fullname, "Nimbus Mono L Regular")) fontname = "Courier";
134     if(!strcmp(fullname, "Nimbus Mono L Regular Oblique")) fontname = "CourierItalic";
135     if(!strcmp(fullname, "Nimbus Mono L Bold")) fontname = "CourierBold";
136     if(!strcmp(fullname, "Nimbus Mono L Bold Oblique")) fontname = "CourierBoldItalic";
137     if(!strcmp(fullname, "Standard Symbols L")) fontname = "Symbol";
138
139     char ** encoding = winAnsiEncoding;
140     int encodingsize = winAnsiEncodingSize;
141
142     printf("processing \"%s\" (\"%s\")...\n", fullname, fontname);
143
144     if(strstr(fullname, "Dingbats")) {// Zapf Dingbats
145         encoding = zapfDingbatsEncoding;
146         encodingsize = zapfDingbatsEncodingSize;
147     }
148     else if(strstr(fullname, "Symbol")) {// Symbol
149         encoding = symbolEncoding;
150         encodingsize = symbolEncodingSize;
151     }
152
153     SWFFONT * wfont = (SWFFONT*)malloc(sizeof(SWFFONT));
154     SWFFont * font = new SWFFont("", i, "");
155
156     memset(wfont, 0, sizeof(SWFFONT));
157
158     wfont->version = 2;
159     wfont->name = (U8*)strdup(fontname);
160     wfont->layout = (SWFLAYOUT*)malloc(sizeof(SWFLAYOUT));
161     memset(wfont->layout, 0, sizeof(SWFLAYOUT));
162
163     int s,num;
164     num = 0;
165     for(s=0;s<encodingsize;s++)
166     {
167         char charinfont=0;
168         if(encoding[s]) {
169             T1_OUTLINE*outline = font->getOutline(encoding[s], 0);
170             if(outline) {num++;charinfont=1;}
171         }
172         /*if(charinfont) printf("x");
173         else printf(".");
174         if((s&15) == 15)
175             printf("\n");*/
176     }
177
178     wfont->maxascii = encodingsize;
179     wfont->numchars = num;
180     
181     wfont->style = (/*bold*/0?FONT_STYLE_BOLD:0) + (angle>0.05?FONT_STYLE_ITALIC:0);
182
183     wfont->glyph = (SWFGLYPH*)malloc(num*sizeof(SWFGLYPH));
184     memset(wfont->glyph, 0, num*sizeof(SWFGLYPH));
185     wfont->glyph2ascii = (U16*)malloc(num*sizeof(U16));
186     memset(wfont->glyph2ascii, 0, num*sizeof(U16));
187     wfont->ascii2glyph = (int*)malloc(encodingsize*sizeof(int));
188     memset(wfont->ascii2glyph, -1, encodingsize*sizeof(int));
189     wfont->layout->ascent = (U16)(underline - bbox.lly);
190     wfont->layout->descent = (U16)(bbox.ury - underline);
191     wfont->layout->leading = (U16)(wfont->layout->ascent - 
192                              wfont->layout->descent -
193                              (bbox.lly - bbox.ury));
194     wfont->layout->bounds = (SRECT*)malloc(sizeof(SRECT)*num);
195     memset(wfont->layout->bounds, 0, sizeof(SRECT)*num);
196     wfont->layout->kerningcount = 0;
197     wfont->layout->kerning = 0;
198   
199     num = 0;
200     for(s=0;s<encodingsize;s++)
201     {
202         if(encoding[s]) {
203             T1_OUTLINE*outline = font->getOutline(encoding[s],0);
204             int width = font->getWidth(encoding[s]);
205
206             if(outline) {
207                 int log = 0;
208                 wfont->ascii2glyph[s] = num;
209                 wfont->glyph2ascii[num] = s;
210                 swf_ShapeNew(&wfont->glyph[num].shape);
211                 SHAPE*shape = wfont->glyph[num].shape;
212                 int firstx = outline->dest.x/0xffff;
213                 
214                 TAG*tag = swf_InsertTag(0,ST_DEFINESHAPE);
215
216                 swfmatrix m;
217                 m.m11 = 1.0;
218                 m.m22 = 1.0;
219                 m.m21 = 0;
220                 m.m12 = 0;
221                 m.m13 = 0;
222                 m.m23 = 0;
223
224                 swf_SetU8(tag,0);
225
226                 shape->bits.fill = 1;
227                 shape->bits.line = 0;
228                 swf_ShapeSetStyle(tag,shape,0,1,0);
229                 resetdrawer();
230
231                 drawpath(tag, outline, &m, log);
232
233                 /*uncomment this to mark the glyph sizes:
234                 plotxy p1,p2; p1.x=0; p1.y=0; p2.x=width/8; p2.y=-width/8;
235                 moveto(tag, p1); lineto(tag, p2); p1.x += 2; p2.x += 2;
236                 lineto(tag, p2); lineto(tag, p1); p1.x -= 2; lineto(tag, p1);// */
237
238                 swf_ShapeSetEnd(tag);
239
240                 wfont->glyph[num].shape->bitlen = (tag->len-1)*8;
241                 wfont->glyph[num].shape->data = (U8*)malloc(tag->len-1);
242                 memcpy(wfont->glyph[num].shape->data, &tag->data[1], tag->len-1);
243                 swf_DeleteTag(tag);
244
245                 /* fix bounding box */
246                 SHAPE2*shape2;
247                 SRECT bbox;
248                 shape2 = swf_ShapeToShape2(shape);
249                 if(!shape2) { fprintf(stderr, "Shape parse error\n");exit(1);}
250                 bbox = swf_GetShapeBoundingBox(shape2);
251                 swf_Shape2Free(shape2);
252                 wfont->layout->bounds[num] = bbox;
253                 //wfont->glyph[num].advance = (int)(width/6.4); // 128/20
254                 wfont->glyph[num].advance = bbox.xmax/20;
255                 if(!wfont->glyph[num].advance) {
256                     wfont->glyph[num].advance = firstx;
257                 }
258                 
259                 num++;
260             }
261         }
262     }
263     return wfont;
264 }
265
266 int main(int argc, char ** argv)
267 {
268   char cwd[128];
269   getcwd(cwd, 128);
270   processargs(argc, argv);
271   if(!all && !filenum) {
272       fprintf(stderr, "You must supply a filename.\n");
273       exit(1);
274   }
275   //TODO: use tempnam here. Check if environment already contains a
276   //T1LIB_CONFIG.
277   putenv( "T1LIB_CONFIG=/tmp/t1lib.config.tmp");
278   FILE*fi = fopen("/tmp/t1lib.config.tmp", "wb");
279   if(all) {
280       fprintf(fi, "FONTDATABASE=/tmp/FontDataBase\n");
281       fprintf(fi, "ENCODING=%s/fonts:.\n", SWFTOOLS_DATADIR);
282       fprintf(fi, "AFM=%s/fonts:.\n", SWFTOOLS_DATADIR);
283       fprintf(fi, "TYPE1=%s/fonts:.\n", SWFTOOLS_DATADIR);
284   } else {
285       fprintf(fi, "FONTDATABASE=/tmp/FontDataBase\n");
286       fprintf(fi, "ENCODING=%s:.\n", cwd);
287       fprintf(fi, "AFM=%s:.\n", cwd);
288       fprintf(fi, "TYPE1=%s:.\n", cwd);
289   }
290   fclose(fi);
291   fi = fopen("/tmp/FontDataBase", "wb");
292   if(all) {
293       fprintf(fi, "14\n");             
294       fprintf(fi, "%s/fonts/n021003l.afm\n", SWFTOOLS_DATADIR);
295       fprintf(fi, "%s/fonts/n021023l.afm\n", SWFTOOLS_DATADIR);
296       fprintf(fi, "%s/fonts/n021004l.afm\n", SWFTOOLS_DATADIR);
297       fprintf(fi, "%s/fonts/n021024l.afm\n", SWFTOOLS_DATADIR);
298       fprintf(fi, "%s/fonts/n019003l.afm\n", SWFTOOLS_DATADIR);
299       fprintf(fi, "%s/fonts/n019023l.afm\n", SWFTOOLS_DATADIR);
300       fprintf(fi, "%s/fonts/n019004l.afm\n", SWFTOOLS_DATADIR);
301       fprintf(fi, "%s/fonts/n019024l.afm\n", SWFTOOLS_DATADIR);
302       fprintf(fi, "%s/fonts/n022003l.afm\n", SWFTOOLS_DATADIR);
303       fprintf(fi, "%s/fonts/n022023l.afm\n", SWFTOOLS_DATADIR);
304       fprintf(fi, "%s/fonts/n022004l.afm\n", SWFTOOLS_DATADIR);
305       fprintf(fi, "%s/fonts/n022024l.afm\n", SWFTOOLS_DATADIR);
306       fprintf(fi, "%s/fonts/s050000l.afm\n", SWFTOOLS_DATADIR);
307       fprintf(fi, "%s/fonts/d050000l.afm\n", SWFTOOLS_DATADIR);
308   } else {
309     fprintf(fi, "%d\n", filenum);
310     int t;
311     for(t=0;t<filenum;t++) {
312         fprintf(fi, "%s/%s\n", cwd, filenames[t]);
313     }
314   }
315   fclose(fi);
316   /* initialize t1lib */
317   T1_SetBitmapPad( 16);
318   if ((T1_InitLib(NO_LOGFILE)==NULL)){
319       fprintf(stderr, "Initialization of t1lib failed\n");
320       exit(1);
321   }
322   unlink("/tmp/t1lib.config.tmp");
323
324   int i,num;
325   for( i=0; i<T1_Get_no_fonts(); i++)
326   {
327     SWFFONT * font = t1font2swffont(i);
328     
329     char filename[128];
330     sprintf(filename, "%s.swf", font->name);
331     swf_WriteFont(font, filename);
332     swf_FontFree(font);
333   }
334   unlink("/tmp/FontDataBase");
335 }
336
337