Merge CV, ver=5.1; ops/methods from HV, and interface from CV where possible
[goodguy/history.git] / cinelerra-5.1 / libzmpeg3 / xfont.C
diff --git a/cinelerra-5.1/libzmpeg3/xfont.C b/cinelerra-5.1/libzmpeg3/xfont.C
new file mode 100644 (file)
index 0000000..70f411e
--- /dev/null
@@ -0,0 +1,393 @@
+#include <stdio.h>
+#include <stdint.h>
+#include <stdlib.h>
+#include <string.h>
+#include <X11/Xutil.h>
+#include <X11/Xlib.h>
+#include "libzmpeg3.h"
+
+/* use builtin data, or dynamically load font data from X instead */
+//#define BUILTIN_FONT_DATA
+/* generate static font data code */
+//#define WRITE_FONT_DATA
+#define MAX_SIZE 4
+
+#ifdef WRITE_FONT_DATA
+#undef BUILTIN_FONT_DATA
+#endif
+
+int zbitfont_t::font_refs = 0;
+
+
+static inline int
+font_index(int style, int pen_size, int italics, int size)
+{
+  if( pen_size > 0 ) --pen_size;
+  return ((size*2 + italics)*2 + pen_size)*7 + style;
+}
+
+#if defined(BUILTIN_FONT_DATA)
+/* using static builtin font data */
+
+zbitfont_t *zmpeg3_t::
+bitfont(int style,int pen_size,int italics,int size)
+{
+  int k = font_index(style,pen_size,italics,size);
+  return zbitfont_t::fonts[k];
+}
+
+zbitfont_t *zwin_t::
+chr_font(zchr_t *bp, int scale)
+{
+  int size = bp->font_size + scale-1;
+  return bitfont(bp->font_style,bp->pen_size,bp->italics,size);
+}
+
+/* stubs */
+void zbitfont_t::init_fonts() {}
+void zbitfont_t::destroy_fonts() {}
+
+#else
+/* loading font data on the fly */
+
+static int
+init_xfont(zbitfont_t *font,char *font_name)
+{
+  const char *display_name = ":0";
+  Display *display = XOpenDisplay(display_name);
+  if( !display ) return 1;
+  Font font_id = XLoadFont(display,font_name);
+  int ret = 0;
+  XFontStruct *font_struct = XQueryFont(display,font_id);
+  if( !font_struct ) ret = 1;
+  if( !ret ) {
+    int row_first = font_struct->min_char_or_byte2;
+    int row_last  = font_struct->max_char_or_byte2;
+    int row_count  = row_last-row_first + 1;
+    int col_first = font_struct->min_byte1;
+    int col_last  = font_struct->max_byte1;
+    XCharStruct *per_char = font_struct->per_char;
+    XCharStruct *bounds = per_char == NULL ? NULL : &font_struct->max_bounds;
+    int row_sz, sizeof_data;
+    Screen *screen = DefaultScreenOfDisplay(display);
+    int depth = DefaultDepthOfScreen(screen);
+    Visual *visual = DefaultVisualOfScreen(screen);
+    Window w = DefaultRootWindow(display);
+    font->nch = (col_last-col_first+1) * (row_last-row_first+1);
+    font->chrs = new zbitchar_t[font->nch];
+    font->idx = font->idy = font->imy = 0;
+    int inx = 0, iny = 0;
+  
+    XImage *image;
+    Pixmap pixmap;
+    int b1, b2, x, y, n;
+    unsigned int width, height;
+    GC gcb, gcw;
+    XGCValues gcv;
+  
+    gcv.function = GXcopy;
+    gcv.line_width = 1;
+    gcv.line_style = LineSolid;
+    gcv.font = font_id;
+    n = GCFunction|GCForeground|GCLineWidth|GCLineStyle|GCFont;
+    gcv.foreground = BlackPixelOfScreen(DefaultScreenOfDisplay(display));
+    gcb = XCreateGC(display,w,n,&gcv);
+    gcv.foreground = WhitePixelOfScreen(DefaultScreenOfDisplay(display));
+    gcw = XCreateGC(display,w,n,&gcv);
+  
+    for( b1=col_first; b1<=col_last; ++b1 ) {
+      for( b2=row_first; b2<=row_last; ++b2 ) {
+        int i = (b1-col_first)*row_count + b2-row_first;
+        zbitchar_t *cp = &font->chrs[i];
+        XChar2b ch;
+        ch.byte1 = b1;
+        ch.byte2 = b2;
+        cp->ch = (b1<<8) + b2;
+        if( per_char != NULL )
+          bounds = &per_char[i];
+        cp->left = bounds->lbearing;
+        cp->right = bounds->rbearing;
+        cp->ascent = bounds->ascent;
+        cp->decent = bounds->descent;
+        width  = cp->right - cp->left;
+        height = cp->ascent + cp->decent;
+        if( cp->ascent > font->imy ) font->imy = cp->ascent;
+        if( cp->decent > iny ) iny = cp->decent;
+        row_sz = (width+7) / 8;
+        sizeof_data = row_sz * height;
+        if( sizeof_data > 0 ) {
+          char *data = (char*)malloc(sizeof_data);
+          if( width > 0 ) { font->idx += width;  ++inx; }
+          memset(&data[0],0,sizeof_data);
+          x = -cp->left;  y = cp->ascent;
+          pixmap = XCreatePixmap(display,w,width,height,depth);
+          XFillRectangle(display,pixmap,gcb,0,0,width,height);
+          XDrawString16(display,pixmap,gcw,x,y,&ch,1);
+          image = XCreateImage(display,visual,1,XYBitmap,0,
+             data,width,height,8,row_sz);
+          XGetSubImage(display,pixmap,0,0,width,height,1,
+             XYPixmap,image,0,0);
+          cp->bitmap = new uint8_t [sizeof_data];
+          memcpy(&cp->bitmap[0],&data[0],sizeof_data);
+          XFreePixmap(display,pixmap);
+          XDestroyImage(image);
+        }
+        else
+          cp->bitmap = 0;
+      }
+    }
+    XFreeGC(display,gcb);
+    XFreeGC(display,gcw);
+    XFreeFont(display,font_struct);
+    if( inx == 0 ) inx = 1;
+    font->idx = (font->idx + inx-1) / inx;
+    font->idy = font->imy + iny;
+  }
+  XCloseDisplay(display);
+  return ret;
+}
+
+static zbitfont_t *
+get_xfont(int style, int pen_size, int italics, int size)
+{
+  int serif = 0, prop = 0, casual = 0; // fst_default
+  switch( style ) {
+  case 1: serif = 1;  prop = 0;  casual = 0;  break; // fst_mono_serif
+  case 2: serif = 1;  prop = 1;  casual = 0;  break; // fst_prop_serif
+  case 3: serif = 0;  prop = 0;  casual = 0;  break; // fst_mono_sans
+  case 4: serif = 0;  prop = 1;  casual = 0;  break; // fst_prop_sans
+  case 5: serif = 1;  prop = 1;  casual = 1;  break; // fst_casual
+  }
+
+  const char *family, *slant;
+#if 1
+  if( casual ) {
+    family = "adobe-utopia";
+    slant = italics ? "i" : "r";
+  }
+  if( serif == 0 ) {
+   family = prop == 0 ?
+     (slant = italics ? "o" : "r", "urw-nimbus mono l") :
+     (slant = italics ? "i" : "r", "urw-nimbus sans l");
+  }
+  else {
+   family = prop == 0 ?
+     (slant = italics ? "i" : "r", "bitstream-courier 10 pitch"):
+     (slant = italics ? "i" : "r", "bitstream-bitstream charter");
+  }
+#else
+  if( casual ) {
+    family = "bitstream-bitstream vera serif";
+    slant = "r";
+  }
+  if( serif == 0 ) {
+   family = prop == 0 ?
+     (slant = italics ? "o" : "r", "bitstream-bitstream vera sans mono") :
+     (slant = italics ? "o" : "r", "bitstream-bitstream vera sans");
+  }
+  else {
+   family = prop == 0 ?
+     (slant = italics ? "i" : "r", "bitstream-courier 10 pitch"):
+     (slant = italics ? "i" : "r", "bitstream-bitstream charter");
+  }
+#endif
+  const char *wght = pen_size > 1 ? "bold" : "medium";
+
+  // range expanded to account for scale
+  if( size < 0 ) size = 0;
+  if( size > MAX_SIZE ) size = MAX_SIZE;
+  int ptsz = 180;
+  switch( size ) {
+  case 0: ptsz =  48;  break; // very fsz_small
+  case 1: ptsz = 100;  break; // fsz_small
+  default:  case 2:    break; // fsz_normal
+  case 3: ptsz = 320;  break; // fsz_large
+  case 4: ptsz = 420;  break; // very fsz_large
+  };
+
+  char name[512];
+  sprintf(&name[0],"-%s-%s-%s-*-*-*-0-%d-%d-*-*-iso8859-1",
+    family, wght, slant, ptsz, ptsz);
+  zbitfont_t *font = new zbitfont_t();
+  if( init_xfont(font,&name[0]) ) {
+    delete font;  font = 0;
+  }
+  return font;
+}
+
+/* italics=2, pen_size=2, font_size=5, font_style=7 */
+int zbitfont_t::total_fonts = 140;
+zbitfont_t *zbitfont_t::fonts[140];
+
+zbitfont_t *zmpeg3_t::
+bitfont(int style,int pen_size,int italics,int size)
+{
+  int k = font_index(style,pen_size,italics,size);
+  bitfont_t *font = bitfont_t::fonts[k];
+  if( !font ) {
+    font = get_xfont(style,pen_size,italics,size);
+    bitfont_t::fonts[k] = font;
+  }
+  return font;
+}
+
+zbitfont_t *zwin_t::
+chr_font(zchr_t *bp, int scale)
+{
+  int size = bp->font_size + scale-1;
+  return bitfont(bp->font_style,bp->pen_size,bp->italics,size);
+}
+
+#if 0
+void zbitfont_t::
+init_fonts()
+{
+  if( !font_refs )
+    for( int i=0; i<total_fonts; ++i ) fonts[i] = 0;
+  ++font_refs;
+}
+
+void zbitfont_t::
+destroy_fonts()
+{
+  if( font_refs > 0 ) --font_refs;
+  if( font_refs > 0 ) return;
+  for( int i=0; i<total_fonts; ++i )
+    if( fonts[i] ) { delete fonts[i];  fonts[i] = 0; }
+}
+#else
+void zbitfont_t::init_fonts() {}
+void zbitfont_t::destroy_fonts() {}
+
+zstatic_init_t::
+static_init_t()
+{
+  for( int i=0; i<total_fonts; ++i ) fonts[i] = 0;
+}
+zstatic_init_t::
+~static_init_t()
+{
+  for( int i=0; i<total_fonts; ++i )
+    if( fonts[i] ) { delete fonts[i];  fonts[i] = 0; }
+}
+
+zstatic_init_t static_init;
+
+#endif
+#endif
+
+#if defined(WRITE_FONT_DATA)
+/* generate code for builtin font initialization */
+
+static int
+write_font(int fno, zbitfont_t *font)
+{
+  char font_var[512];
+  
+  sprintf(&font_var[0],"font%02d",fno);
+  printf("#include \"libzmpeg3.h\"\n\n");
+  zbitchar_t *cp = font->chrs;
+  for( int i=0; i<font->nch; ++i,++cp ) {
+    int sizeof_data = (cp->right-cp->left+7)/8 * (cp->ascent+cp->decent);
+    if( sizeof_data > 0 ) {
+      printf("static uint8_t %s_pix%04x[] = {\n",font_var,i);
+      uint8_t *bp = cp->bitmap;
+      int j = 0;
+      while( j < sizeof_data ) {
+        if( (j&7) == 0 ) printf("  ");
+        printf("0x%02x,",*bp++);
+        if( (++j&7) == 0 ) printf("\n");
+      }
+      if( (j&7) != 0 ) printf("\n");
+      printf("};\n");
+    }
+  }
+  printf("\n");
+
+  printf("static zbitchar_t %s_chrs[] = {\n",font_var);
+  cp = font->chrs;
+  for( int i=0; i<font->nch; ++i,++cp ) {
+    printf("  { 0x%04x, %2d, %2d, %2d, %2d, ",
+      cp->ch,cp->left,cp->right,cp->ascent,cp->decent);
+    int sizeof_data = (cp->right-cp->left+7)/8 * (cp->ascent+cp->decent);
+    if( sizeof_data > 0 )
+      printf("&%s_pix%04x[0] },\n",font_var,i);
+    else
+      printf("0 },\n");
+  }
+  printf("};\n");
+  printf("\n");
+  
+  printf("static zbitfont_t %s = { &%s_chrs[0], %d, %d, %d, %d, };\n",
+    font_var,font_var,font->nch,font->idx,font->idy,font->imy);
+  printf("\n");
+  return 0;
+}
+
+int main(int ac, char **av)
+{
+  int fno = 0;
+  for( int size=0; size<=MAX_SIZE; ++size ) {
+    for( int italics=0; italics<=1; ++italics ) {
+      for( int pen_size=1; pen_size<=2; ++pen_size ) {
+        for( int font_style=0; font_style<=6; ++font_style ) {
+          zbitfont_t *font = get_xfont(font_style, pen_size, italics, size);
+          write_font(++fno, font);
+          delete font;
+        }
+      }
+    }
+  }
+  fno = 0;
+  printf("zbitfont_t *zbitfont_t::fonts[] = {\n");
+  for( int size=0; size<=MAX_SIZE; ++size ) {
+    for( int italics=0; italics<=1; ++italics ) {
+      for( int pen_size=1; pen_size<=2; ++pen_size ) {
+        for( int font_style=0; font_style<=6; ++font_style ) {
+          if( (fno&7) == 0 ) printf("  ");
+          printf("&font%02d,",++fno);
+          if( (fno&7) == 0 ) printf("\n");
+        }
+      }
+    }
+  }
+  if( (fno&7) != 0 ) printf("\n");
+  printf("};\n");
+  printf("\n");
+  printf("int zbitfont_t::total_fonts = %d;\n",fno);
+  printf("\n");
+  return 0;
+}
+
+#endif
+
+#if 0
+/* attach to end of output for testing */
+int main(int ac, char **av)
+{
+  int sz = 0;
+  for( int i=0; i<zbitfont_t::total_fonts; ++i ) {
+    zbitfont_t *font = zbitfont_t::fonts[i];
+    for( int ch=0; ch<font->nch; ++ch ) {
+      zbitchar_t *cp = &font->chrs[ch];
+      sz += 16 + (cp->ascent+cp->decent)*((cp->right-cp->left+7)/8);
+    }
+    sz += 20;
+  }
+  printf(" %d bytes\n",sz);
+  zbitchar_t *cp = &font84.chrs['y'];
+  uint8_t *bmap = cp->bitmap;
+  int wd = cp->right - cp->left;
+  int ht = cp->ascent + cp->decent;
+  int wsz = (wd+7) / 8;
+  for( int iy=0; iy<ht; ++iy ) {
+    uint8_t *row = &bmap[iy*wsz];
+    for( int ix=0; ix<wd; ++ix ) {
+      int ch = ((row[ix>>3] >> (ix&7)) & 1) ?  'x' : ' ';
+      printf("%c",ch);
+    }
+    printf("\n");
+  }
+  return 0;
+}
+#endif