/** \file
 * \brief DGN driver
 *
 * See Copyright Notice in cd.h
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <math.h>
#include <limits.h>

#include "cd.h"
#include "cd_private.h"
#include "cddgn.h"

/* defines */

#define MAX_NUM_VERTEX 101
#define MAX_NUM_VERTEX_PER_POLYLINE 15000

#ifndef PI
#define PI 3.14159265358979323846
#endif

#define END_OF_DGN_FILE 0xffff
#define DGN_FILE_BLOCK  512

#define NOFILL 0    /* tipos de fill que o driver faz */
#define CONVEX 1
#define NORMAL 2 

/* macros */

#define SIZE_LINE_STRING(x)  (19+4*(x))
#define SIZE_FILLED_SHAPE(x) (27+4*(x))
#define SIZE_ARC             40
#define SIZE_LINE            26

#define IGNORE(x) ((void) x)


/* estruturas similares as que o MicroStation usa */

typedef struct
{
  union 
  {
    struct 
    {  
      unsigned level:6;
      unsigned :1;
      unsigned complex:1;
      unsigned type:7;
      unsigned :1;
    } flags;
    short type_as_word;
  } type;

  unsigned short words;
  unsigned long  xmin;
  unsigned long  ymin;
  unsigned long  xmax;
  unsigned long  ymax;
} Elm_hdr; 

typedef struct
{
  short attindx;
  union
  {
    short s;
    struct
    {
      unsigned /*class*/ :4;
      unsigned /*res*/   :4;
      unsigned /*l*/     :1;
      unsigned /*n*/     :1;
      unsigned /*m*/     :1;
      unsigned attributes:1;
      unsigned /*r*/     :1;
      unsigned /*p*/     :1;
      unsigned /*s*/     :1;
      unsigned /*hole*/  :1;
    } flags;
  } props;

  union
  {
    short s;
    struct
    {
      unsigned style:3;
      unsigned weight:5;
      unsigned color:8;
    } b;
  } symb;

} Disp_hdr;

/* tipo de poligono/setor */
enum
{
  FILLED,
  OPEN 
};

/* grupos de tamanhos de caracter 
   (usado por gettextwidth e cdgetfontdim) */

static long fontsizes[4][8]=
{
  {1,2,4,5,6,7,0},
  {5,3,2,1,0},
  {8,6,4,3,2,1,0},
  {5,3,2,1,0}
};


/**********************
 * contexto do driver *
 **********************/

struct _cdCtxCanvas
{
  cdCanvas* canvas;

  FILE *file;                             /* arquivo dgn */
  long int bytes;                            /* tamanho do arquivo */
  char level;
  
  short color, style;                 

  short alignment;
  short typeface_index;
  short symbology;
  long tl;
  short is_base;                               /* setado se texto e' do tipo CD_BASE_...   */

  short fill_type;                   /* como o driver faz fill:
                                           NOFILL -> nao faz fill
                                           CONVEX -> so faz fill de poligonos convexos
                                           NORMAL -> faz fill normalmente */

  long colortable[256];              /* palette */
  short num_colors;

  short is_complex;
};

/* prototipos de funcao */
static void startComplexShape(cdCtxCanvas*, unsigned short,short,unsigned short,
                            unsigned long,unsigned long,
                            unsigned long,unsigned long);
static void endComplexElement(cdCtxCanvas*);

/******************************
 *                            *
 * funcoes internas do driver *
 *                            *
 ******************************/

/*********************************************************
 * Obtem o descent do texto (para letras como q,p,g etc) * 
 *********************************************************/

static long get_descent(const char *text, int len, int size_pixel)
{
 char *descent="jgyqp";
 long  a=0; 

 while(a < len)
 {
   if(strchr(descent, text[a]))
     return size_pixel/2;

   a++;
 }
 return 0;
}

/***********************************************
 * Calcula a largura da string no MicroStation *
 ***********************************************/

static long gettextwidth(cdCtxCanvas* ctxcanvas, const char *s, int len, int size_pixel)
{
  long a=0,
       width=0;

  short default_size=0;

  static char *fontchars[4][8] =
  {
    {         /* CD_SYSTEM */
      "Ww",
      "jshyut#*-=<>",
      "iIl[]",
      ";:.,'|!()`{}",
      "","","",""
    }, 

    { /* CD_COURIER */
      "Iflrit!();.'",
      "1|[]\"/`",
      "BCDEKPRSUVXYbdgkpq&-_", 
      "w#%", 
      "Wm^+=<>~",
      "@","",""
    }, 

    { /* CD_TIMES_ROMAN */
      "m",
      "HMUWw",
      "CSTZLbhknpuvxy23567890e",
      "fstz1#$-=<>", 
      "Iijl*[]", 
      ";:.,'|!()`{}",
      "",""
    },
    
    { /* CD_HELVETICA */
      "Ww",
      "jshyut#*-=<>",
      "iIl[]", 
      ";:.,'|!()`{}",
      "","","",""
    }  
  };

  if (ctxcanvas->typeface_index == 1)
    default_size=2;
  else if (ctxcanvas->typeface_index == 2)
    default_size=5;
  else if (ctxcanvas->typeface_index == 3)
    default_size=4;
  else
    default_size=4;

  for(a=0,width=0;a < len; a++)
  { 
    static short size_number;
    static char letter;
   
    if(s[a] == ' ')
      letter = s[a-1];
    else
      letter = s[a];
      
    for(size_number=0;size_number < 8;size_number++)
    {
      if(strchr(fontchars[ctxcanvas->typeface_index][size_number], letter))
      {
        width+=(ctxcanvas->tl*fontsizes[ctxcanvas->typeface_index][size_number])/6;
        break;
      }
    }
    
    if(size_number == 8)
      width+=(ctxcanvas->tl*default_size)/6;

    width+=ctxcanvas->tl/3;
  }
  
  width-=ctxcanvas->tl/3;

  if (ctxcanvas->canvas->font_style & CD_ITALIC)
    width+= (long) ((double)size_pixel*tan(4*atan(1)/8)); /* angulo de 15 graus */

  return width;
}

/****************************
 * Salva um byte no arquivo *
 ****************************/

static void put_byte(cdCtxCanvas* ctxcanvas, unsigned char byte)
{
  fputc(byte, ctxcanvas->file);
}

/************************************
 * Salva um sequencia de caracteres *
 ************************************/

static void writec (cdCtxCanvas* ctxcanvas, const char *t, short tam )
{
 short i;

 ctxcanvas->bytes += tam;
 for ( i = 0; i < tam; i++ )
     fputc ( t[i], ctxcanvas->file );
}

/******************
 * Salva uma word *
 ******************/

static void put_word(cdCtxCanvas* ctxcanvas, unsigned short w)
{
  char c;

  c = (char) (w & 0xff);
  fputc (c, ctxcanvas->file);
  c = (char) ((w >> 8) & 0xff);
  fputc (c, ctxcanvas->file);
  ctxcanvas->bytes += 2;
}

/****************************
 * Salva um long no arquivo *
 ****************************/

static void put_long (cdCtxCanvas* ctxcanvas, unsigned long i)
{
  put_word(ctxcanvas, (short) (i >> 16));
  put_word(ctxcanvas, (short) i);
}

/*******************
 * Salva um double *
 *******************/

static void put_as_double(cdCtxCanvas* ctxcanvas, long i)
{ 
  float dfloat=(float) 4*i;
 
  put_long(ctxcanvas, *((long *) &dfloat)); 
  put_long(ctxcanvas, 0);

  ctxcanvas->bytes+=sizeof(float);
}

/****************************
 * Salva uma UOR no arquivo *
 ****************************/

static void put_uor(cdCtxCanvas* ctxcanvas, long i)
{
  put_word(ctxcanvas, (unsigned short)((i >> 16) ^ (1 << 15))); /* troca o bit 31 
                                             para transformar em uor */
  put_word(ctxcanvas, (unsigned short)i);
}

/*************************************** 
 * Salva a bounding box de um elemento *
 ***************************************/

static void put_bound(cdCtxCanvas* ctxcanvas, long xmin, long xmax, long ymin, long ymax)
{
  put_uor(ctxcanvas, xmin);
  put_uor(ctxcanvas, ymin);
  put_uor(ctxcanvas, 0L);
  put_uor(ctxcanvas, xmax);
  put_uor(ctxcanvas, ymax);
  put_uor(ctxcanvas, 0L);
}

/******************************************
 * Calcula a bounding box de uma polyline *
 ******************************************/

static void line_string_bound(cdPoint *buffer, short num_vertex,
                     unsigned long *xmin, unsigned long *ymin, 
                     unsigned long *xmax, unsigned long *ymax)
{
  short i;
  unsigned long v;
  
  *xmin = *xmax = buffer[0].x;
  *ymin = *ymax = buffer[0].y;

  for (i = 1; i < num_vertex; i++)
  {
	  v = buffer[i].x;
    if (v < *xmin)
      *xmin = v;
    else if (v > *xmax)
      *xmax = v;

   v = (long) buffer[i].y;
	 if (v < *ymin)
      *ymin = v;
    else if (v > *ymax)
      *ymax = v;
  }
}

/************************************
 * Retorna symbology de um elemento *
 ************************************/

static short symbology(cdCtxCanvas* ctxcanvas)
{
  return (short)((ctxcanvas->color << 8) | (ctxcanvas->canvas->line_width << 3) | ctxcanvas->style);
} 

/*****************************************
 * Salva um Element Header no arquivo *
 *****************************************/

static void putElementHeader(cdCtxCanvas* ctxcanvas, Elm_hdr *ehdr)
{
  ehdr->type.flags.complex = ctxcanvas->is_complex;
  
  put_word(ctxcanvas, (short)(ehdr->type.flags.type << 8 |
    ehdr->type.flags.complex << 7 | ehdr->type.flags.level));
  

  put_word(ctxcanvas, ehdr->words);  
  put_bound(ctxcanvas, ehdr->xmin, ehdr->xmax, ehdr->ymin, ehdr->ymax);
}

/**************************************
 * Salva um display header no arquivo *
 **************************************/

static void putDisplayHeader(cdCtxCanvas* ctxcanvas, Disp_hdr *dhdr)
{
  put_word(ctxcanvas, 0);                      /* graphics group */
  put_word(ctxcanvas, dhdr->attindx);          /* index to attributes */
  put_word(ctxcanvas, dhdr->props.flags.attributes << 11); /* properties */
  put_word(ctxcanvas, dhdr->symb.s);           /* display symbology */
}


/***************************************
 * completa o arquivo com zeros para   *
 * que o numero de bytes seja multiplo *
 * de 512                              *
 ***************************************/

static void complete_file(cdCtxCanvas* ctxcanvas)
{
  long resto, i;

  put_word(ctxcanvas, END_OF_DGN_FILE);

  resto = DGN_FILE_BLOCK - ctxcanvas->bytes % DGN_FILE_BLOCK;

  /* checa validade do tamanho do arquivo */
  if (resto%2 != 0) return;

  for (i = 0; i < resto; i+=2)
    put_word(ctxcanvas, 0);
}

/*************************************
 * Salva um elemento arco no arquivo *
 *************************************/

static void arc (cdCtxCanvas* ctxcanvas, long xc, long yc, long w, long h, double a1, double a2)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr; 

    /* raster header element */
  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=16;
  ehdr.words=SIZE_ARC-2;
  ehdr.xmin=xc - w/2;
  ehdr.xmax=xc + w/2;
  ehdr.ymin=yc - h/2;
  ehdr.ymax=yc + h/2;

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx = ehdr.words - 14;
  dhdr.props.flags.attributes = 0;
  dhdr.symb.s = symbology(ctxcanvas);
  putDisplayHeader(ctxcanvas, &dhdr);

  put_long(ctxcanvas, (long) a1*360000);       /* start angle */
  put_long(ctxcanvas, (long) (a2-a1)*360000);  /* sweep angle */
  put_as_double(ctxcanvas, w/2);           /* primary axis */
  put_as_double(ctxcanvas, h/2);           /* secondary axis */
  put_long(ctxcanvas, 0);                      /* rotation angle (sempre 0) */
  put_as_double(ctxcanvas, xc);          /* x origin */
  put_as_double(ctxcanvas, yc);          /* y origin */
}

/***************************************
 * Salva um elemento elipse no arquivo *
 ***************************************/

static void ellipse(cdCtxCanvas* ctxcanvas, long xc, long yc, long w, long h, short type)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr; 

    /* raster header element */
  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=15;
  ehdr.words=34+((type==FILLED) ? 8 : 0);
  ehdr.xmin=xc - w/2;
  ehdr.xmax=xc + w/2;
  ehdr.ymin=yc - h/2;
  ehdr.ymax=yc + h/2;

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=20;
  dhdr.props.flags.attributes=(type == FILLED) ? 1 : 0;
  dhdr.symb.s=symbology(ctxcanvas);
  putDisplayHeader(ctxcanvas, &dhdr);

  put_as_double(ctxcanvas, w/2);           /* primary axis */
  put_as_double(ctxcanvas, h/2);           /* secondary axis */
  put_long(ctxcanvas, 50);               /* rotation angle (sempre 0) */
  put_as_double(ctxcanvas, xc);          /* x origin */
  put_as_double(ctxcanvas, yc);          /* y origin */
  
    /* salva atributo de fill */
  if(type == FILLED)
  {
    put_word(ctxcanvas, 0x1007);
    put_word(ctxcanvas, 65);
    put_word(ctxcanvas, 0x802);
    put_word(ctxcanvas, 0x0001);
    put_word(ctxcanvas, ctxcanvas->color);
    put_word(ctxcanvas, 0);
    put_word(ctxcanvas, 0);
    put_word(ctxcanvas, 0);
  }
}

static short getclosestColor(cdCtxCanvas* ctxcanvas, long color)
{
  short count=0, closest=0;
  long diff=0;
  unsigned char r = cdRed(color),
                g = cdGreen(color),
                b = cdBlue(color);
  short rd, gd, bd;
  long newdiff;

  /* procura a cor mais proxima */

  diff = 3*65536;  /* inicializa com maior diferenca possivel */

  for(count=0; count < ctxcanvas->num_colors; count++)
  {
    rd = r - cdRed(ctxcanvas->colortable[count]);
    gd = g - cdGreen(ctxcanvas->colortable[count]);
    bd = b - cdBlue(ctxcanvas->colortable[count]);

    newdiff = rd*rd + gd*gd + bd*bd;

    if(newdiff <= diff)
    {
      /* verifica se encontrou a cor */
      if(newdiff == 0)
        return count-1;

      diff = newdiff;
      closest=count-1;
    }
  }

  /* nao encontrou a cor, tenta adicionar na palette, ou retorna a mais proxima */
  if(ctxcanvas->num_colors < 254)
  {
    ctxcanvas->colortable[ctxcanvas->num_colors+1] = color;
      return ctxcanvas->num_colors++;
  }
  else
    return closest;
}

static void saveColorTable(cdCtxCanvas* ctxcanvas)
{
  unsigned char r,g,b;
  short i;
  
  put_word(ctxcanvas, (0x05 << 8) | 1);  /* colortable */
  put_word(ctxcanvas, 434);

  put_long(ctxcanvas, 0);
  put_long(ctxcanvas, 0);
  put_long(ctxcanvas, 0);
  put_long(ctxcanvas, 0xffffffff);
  put_long(ctxcanvas, 0xffffffff);
  put_long(ctxcanvas, 0xffffffff);

  put_word(ctxcanvas, 0);
  put_word(ctxcanvas, 420);
  put_word(ctxcanvas, 0x0400);
  put_word(ctxcanvas, 0x100);
  put_word(ctxcanvas, 0);

  for(i=0;i<256;i++)
  {
	 cdDecodeColor(ctxcanvas->colortable[i], &r, &g, &b);
	 put_byte(ctxcanvas, r);
	 put_byte(ctxcanvas, g);
	 put_byte(ctxcanvas, b);
  }

  put_word(ctxcanvas, 25);
  for(i=0;i<32;i++)
	 put_word(ctxcanvas, 0x2020);
}


/*****************************
 * Le uma word de um arquivo *
 *****************************/

static short file_get_word(FILE *fp)
{
  short word=0;

  word = (short)fgetc(fp);
  word |= ((short)fgetc(fp) << 8) & 0xff00;    

  return word;
}

/********************************
 * Salva uma word em um arquivo *
 ********************************/

static void file_put_word (short word, FILE *fp)
{
  fputc ((char) (word & 0xff), fp);
  fputc ((char) ((word >> 8) & 0xff), fp);
}

/*******************************************
 * Le elementos de um arquivo DGN e os     *
 * coloca no inicio do arquivo aberto pelo *
 * driver                                  *
 *******************************************/

static void dgn_copy (FILE *file, cdCtxCanvas *ctxcanvas)
{
  short word=0;

  while ((word = file_get_word(file)) != END_OF_DGN_FILE) 
  {
    file_put_word(word, ctxcanvas->file); /* type e level do elemento */
    ctxcanvas->bytes+=2;

    word = file_get_word(file); /* words to follow */
    file_put_word(word, ctxcanvas->file);
    ctxcanvas->bytes+=2;

    while (word)       /* copia resto do elemento */
    {
      file_put_word(file_get_word(file), ctxcanvas->file);
      word--;
      ctxcanvas->bytes+=2;
    }
  }
}


/*
 *  Funcoes do driver
 */

static void cdkillcanvas(cdCtxCanvas* ctxcanvas)
{
  saveColorTable(ctxcanvas);
  complete_file(ctxcanvas);
  fclose (ctxcanvas->file);

  memset(ctxcanvas, 0, sizeof(cdCtxCanvas));
  free(ctxcanvas);
}

static void cddeactivate (cdCtxCanvas* ctxcanvas)
{
  fflush(ctxcanvas->file);
}

static void cdflush (cdCtxCanvas* ctxcanvas)
{
  fflush(ctxcanvas->file);
}


/******************************************************/
/* primitives                                         */
/******************************************************/

static void cdline (cdCtxCanvas* ctxcanvas, int x1, int y1, int x2, int y2)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr;
  cdPoint buffer[2];

  buffer[0].x=x1;
  buffer[0].y=y1;
  buffer[1].x=x2;
  buffer[1].y=y2;
  
  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=3;
  
  ehdr.words=SIZE_LINE-2;

  line_string_bound(buffer, 2, &ehdr.xmin,
                    &ehdr.ymin,&ehdr.xmax,&ehdr.ymax);
  
  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx = ehdr.words - 14;
  dhdr.props.flags.attributes = 0;
  dhdr.symb.s=symbology(ctxcanvas);

  putDisplayHeader(ctxcanvas, &dhdr);

    /* pontos inicial e final da linha */

  put_long(ctxcanvas, (long) x1);     
  put_long(ctxcanvas, (long) y1);
  put_long(ctxcanvas, (long) x2);
  put_long(ctxcanvas, (long) y2);
}

static void cdbox (cdCtxCanvas* ctxcanvas, int xmin, int xmax, int ymin, int ymax)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr; 

    /* raster header element */
  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=6;
  ehdr.words=17+4*5+8;
  ehdr.xmin=xmin;
  ehdr.xmax=xmax;
  ehdr.ymin=ymin;
  ehdr.ymax=ymax;

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=3+4*5;
  dhdr.props.flags.attributes=1;
  dhdr.symb.s=symbology(ctxcanvas);
  putDisplayHeader(ctxcanvas, &dhdr);

  put_word(ctxcanvas, 5);                       /* numero de vertices */

   /* vertices */
  put_long(ctxcanvas, (long) xmin);
  put_long(ctxcanvas, (long) ymin);
  
  put_long(ctxcanvas, (long) xmax);
  put_long(ctxcanvas, (long) ymin);
 
  put_long(ctxcanvas, (long) xmax);
  put_long(ctxcanvas, (long) ymax);

  put_long(ctxcanvas, (long) xmin);
  put_long(ctxcanvas, (long) ymax);

  put_long(ctxcanvas, (long) xmin);
  put_long(ctxcanvas, (long) ymin);
 
    /* atributos de fill */

  put_word(ctxcanvas, 0x1007);
  put_word(ctxcanvas, 65);
  put_word(ctxcanvas, 0x802);
  put_word(ctxcanvas, 0x0001);
  put_word(ctxcanvas, ctxcanvas->color);
  put_word(ctxcanvas, 0);
  put_word(ctxcanvas, 0);
  put_word(ctxcanvas, 0);
}

static void cdarc (cdCtxCanvas* ctxcanvas, int xc, int yc, int w, int h, double a1, double a2)
{
  if (a2 == a1 + 360)
    ellipse(ctxcanvas, xc,yc,w,h,OPEN);
  else
    arc(ctxcanvas, xc, yc, w, h, a1, a2);
}

static void cdsector (cdCtxCanvas* ctxcanvas, int xc, int yc, int w, int h, double a1, double a2)
{
  if (a2 == a1 + 360)
  {
    ellipse(ctxcanvas, xc,yc,w,h,FILLED);
    return;
  }

  startComplexShape(ctxcanvas, 3, 1, SIZE_ARC+SIZE_LINE*2,
    (unsigned long) xc-w/2, (unsigned long) yc-h/2,
    (unsigned long) xc+h/2, (unsigned long) yc+h/2);

  arc(ctxcanvas, xc, yc, w, h, a1, a2);
  
  cdline(ctxcanvas, xc, yc, (int)
    (((double)w*cos(a1*CD_DEG2RAD)/2.+.5)), (int) (((double)h*sin(a1*CD_DEG2RAD))/2.+.5));
  cdline(ctxcanvas, xc, yc, (int) 
    (((double)w*cos(a2*CD_DEG2RAD)/2.+.5)), (int) (((double)h*sin(a2*CD_DEG2RAD))/2.+.5));

  endComplexElement(ctxcanvas);
}

static void cdtext (cdCtxCanvas* ctxcanvas, int x, int y, const char *s, int len)
{
  long descent=0;
  short w=0;
  long hc=0,wc=0;
  int size_pixel;
  short italic = (ctxcanvas->canvas->font_style&CD_ITALIC);
 
  Elm_hdr ehdr;
  Disp_hdr dhdr;

  if(len > 255)
    len=255;

  w = (short)((len/2)+(len%2));
  size_pixel = cdGetFontSizePixels(ctxcanvas->canvas, ctxcanvas->canvas->font_size);
  descent=get_descent(s, len, size_pixel);
  hc = size_pixel+descent;
  wc = gettextwidth(ctxcanvas, s, len, size_pixel);
 
  y+=descent;

  switch (ctxcanvas->alignment)
  {
  case 12: x = x;                   y = y;                   break;
  case 13: x = x;                   y = y - (int) (hc/2.0);  break;
  case 14: x = x;                   y = y - (int) hc;        break;
  case  6: x = x - (int) (wc/2.0);  y = y;                   break;
  case  7: x = x - (int) (wc/2.0);  y = y - (int) (hc/2.0);  break;
  case  8: x = x - (int) (wc/2.0);  y = y - (int) hc;        break;
  case  0: x = x - (int) wc;        y = y;                   break;
  case  1: x = x - (int) wc;        y = y - (int) (hc/2.0);  break;
  case  2: x = x - (int) wc;        y = y - (int) hc;        break;
  }

  if(ctxcanvas->is_base)
    y -= (int) (hc/4.0);

   /* raster header element */
  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=17;
  ehdr.words=28+w+((italic) ? 8 : 0);
  ehdr.xmin=x;
  ehdr.xmax=x+wc;
  ehdr.ymin=y-descent;
  ehdr.ymax=y+hc;

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=14+w;
  dhdr.props.flags.attributes=(italic) ? 1 : 0;

  if (ctxcanvas->canvas->font_style&CD_BOLD)
    dhdr.symb.s=ctxcanvas->color << 8 | (3 << 3);
  else
    dhdr.symb.s=ctxcanvas->color << 8;

  putDisplayHeader(ctxcanvas, &dhdr);

  put_word(ctxcanvas, (ctxcanvas->alignment << 8) | ctxcanvas->typeface_index);

  put_long(ctxcanvas, (long)((1000 * ctxcanvas->tl) / 6) | (1 << 7));
  put_long(ctxcanvas, (long)((1000 * size_pixel) / 6) | (1 << 7));
 
  put_long(ctxcanvas, 0);
  put_long(ctxcanvas, x);
  put_long(ctxcanvas, y);
  put_word(ctxcanvas, (unsigned short)len);
  writec(ctxcanvas, s, (short)(len+(len%2))); /* deve escrever sempre um numero par de caracteres */

  if(italic)
  {
    put_word(ctxcanvas, 0x1007);  /* atributos e words to follow */
    put_word(ctxcanvas, 0x80d4);  /* tipo de atributo */
    put_long(ctxcanvas, 0x000865c0);
    put_long(ctxcanvas, 0x00520000);
    put_long(ctxcanvas, 0x00000000);
  }
}

static void startComplexShape(cdCtxCanvas* ctxcanvas, 
                              unsigned short num_elements,
                              short is_fill,
                              unsigned short size,
                              unsigned long xmin,
                              unsigned long ymin,
                              unsigned long xmax,
                              unsigned long ymax)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr;

  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=14;
  ehdr.words=22 + ((is_fill) ? 8 : 0);
  ehdr.xmax = xmax; 
  ehdr.xmin = xmin; 
  ehdr.ymax = ymax; 
  ehdr.ymin = ymin; 

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=4;
  dhdr.props.flags.attributes = (is_fill) ? 1 : 0;
  dhdr.symb.s=symbology(ctxcanvas);

  putDisplayHeader(ctxcanvas, &dhdr);

  put_word(ctxcanvas, size + 5 + ((is_fill) ? 8 : 0));
  put_word(ctxcanvas, num_elements);

  put_long(ctxcanvas, 0);  /* atributo nulo */
  put_long(ctxcanvas, 0);
 
    /* salva atributo de fill */
  if(is_fill)
  {
    put_word(ctxcanvas, 0x1007);
    put_word(ctxcanvas, 65);
    put_word(ctxcanvas, 0x802);
    put_word(ctxcanvas, 0x0001);
    put_word(ctxcanvas, ctxcanvas->color);
    put_word(ctxcanvas, 0);
    put_word(ctxcanvas, 0);
    put_word(ctxcanvas, 0);
  }

  /* marca inicio de elemento complexo */

  ctxcanvas->is_complex = 1;
}


static void startComplexChain(cdCtxCanvas* ctxcanvas, 
                              unsigned short num_elements,
                              unsigned short size,
                              unsigned long xmin,
                              unsigned long ymin,
                              unsigned long xmax,
                              unsigned long ymax)
                            
{
  Elm_hdr ehdr;
  Disp_hdr dhdr;

  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=12;

  ehdr.words=22;
  ehdr.xmax = xmax; 
  ehdr.xmin = xmin; 
  ehdr.ymax = ymax; 
  ehdr.ymin = ymin; 


  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=4;
  dhdr.props.flags.attributes = 1;
  dhdr.symb.s=symbology(ctxcanvas);

  putDisplayHeader(ctxcanvas, &dhdr);

  put_word(ctxcanvas, size+5);
  put_word(ctxcanvas, num_elements);

  put_long(ctxcanvas, 0);  /* atributo nulo */
  put_long(ctxcanvas, 0);

  
  /* marca inicio de elemento complexo */

  ctxcanvas->is_complex = 1;
}

static void endComplexElement(cdCtxCanvas* ctxcanvas)
{
  ctxcanvas->is_complex = 0;
}

static void putLineString(cdCtxCanvas* ctxcanvas, cdPoint *buffer, short num_vertex)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr;
  short i=0;

  ehdr.type.flags.level=ctxcanvas->level;

  ehdr.type.flags.type=4;
  
  ehdr.words=SIZE_LINE_STRING(num_vertex)-2;

  line_string_bound(buffer, num_vertex, &ehdr.xmin,
                    &ehdr.ymin,&ehdr.xmax,&ehdr.ymax);
  
  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=ehdr.words-14;
  dhdr.props.flags.attributes = 0;
  dhdr.symb.s=symbology(ctxcanvas);

  putDisplayHeader(ctxcanvas, &dhdr);

  put_word(ctxcanvas, num_vertex);

  for (i = 0; i < num_vertex; i++)
  {
    put_long(ctxcanvas, (long) buffer[i].x);
    put_long(ctxcanvas, (long) buffer[i].y);
  }
}

static void putShape(cdCtxCanvas* ctxcanvas, cdPoint *buffer, short num_vertex)
{
  Elm_hdr ehdr;
  Disp_hdr dhdr;
  short i=0;

  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=6;

  ehdr.words=SIZE_FILLED_SHAPE(num_vertex)-2;

  line_string_bound(buffer, num_vertex, &ehdr.xmin,
                    &ehdr.ymin,&ehdr.xmax,&ehdr.ymax);
  

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=ehdr.words - 14 - 8; /* 8 -> size of attributes */
  dhdr.props.flags.attributes = 1;
  dhdr.symb.s=symbology(ctxcanvas);

  putDisplayHeader(ctxcanvas, &dhdr);

  put_word(ctxcanvas, num_vertex);

  for (i = 0; i < num_vertex; i++)
  {
    put_long(ctxcanvas, (long) buffer[i].x);
    put_long(ctxcanvas, (long) buffer[i].y);
  }

  put_word(ctxcanvas, 0x1007);
  put_word(ctxcanvas, 65);
  put_word(ctxcanvas, 0x802);
  put_word(ctxcanvas, 0x0001);
  put_word(ctxcanvas, ctxcanvas->color);
  put_word(ctxcanvas, 0);
  put_word(ctxcanvas, 0);
  put_word(ctxcanvas, 0);
}

static void cdpoly(cdCtxCanvas* ctxcanvas, int mode, cdPoint* poly, int n)
{
  short is_fill=0;

  if (mode == CD_BEZIER)
  {
    cdSimPolyBezier(ctxcanvas->canvas, poly, n);
    return;
  }
  if (mode == CD_PATH)
  {
    cdSimPolyPath(ctxcanvas->canvas, poly, n);
    return;
  }

  if(mode == CD_FILL && ctxcanvas->fill_type == NOFILL)
    mode = CD_CLOSED_LINES;

  if(n > MAX_NUM_VERTEX_PER_POLYLINE)
    n = MAX_NUM_VERTEX_PER_POLYLINE;

  /* acerta buffer de pontos */
  if(mode == CD_FILL || mode == CD_CLOSED_LINES)
  {
    poly[n].x = poly[0].x;
    poly[n].y = poly[0].y;
    n++;
  }

  /* se fill_type for CONVEX, testa se poligono e' convexo ou concavo */
  if((ctxcanvas->fill_type == CONVEX) && (n > 3) && (mode == CD_FILL))
  {
    short signal=0;
    short count=0;
    long vect=0;

    /* calcula sinal do vetorial entre o primeiro lado e o segundo */
    vect = (poly[1].x - poly[0].x) * 
           (poly[2].y - poly[1].y) -
           (poly[1].y - poly[0].y) * 
           (poly[2].x - poly[1].x);

    if(vect == 0)
      mode = CD_CLOSED_LINES;  /* ver se precisa mudar */
    else
    {
      signal = (short)(vect/abs(vect));

      for(count=1 ; count< (n-2); count++)
      { 
        vect = (poly[count+1].x - poly[count].x) * 
               (poly[count+2].y - poly[count+1].y) -
               (poly[count+1].y - poly[count].y) * 
               (poly[count+2].x - poly[count+1].x);
      
        if(vect == 0)
        {
          mode=CD_CLOSED_LINES;
          break;
        }

        if((vect/abs(vect)) != signal)
        {
          mode=CD_CLOSED_LINES;
          break;
        }
      }
    }
  }

  /* se tiver fill */

  if(mode == CD_FILL)
    is_fill=1;

  if(n > MAX_NUM_VERTEX)  /* tem que usar complex shape ou chain */
  {
    short count=0;
    short num_whole_elements = n / MAX_NUM_VERTEX;
    short num_whole_vertex = num_whole_elements * MAX_NUM_VERTEX;
    short rest = n % MAX_NUM_VERTEX;
    short is_there_rest = (rest > 0) ? 1 : 0;
    unsigned long xmax, xmin, ymax, ymin;
    unsigned short size =
         SIZE_LINE_STRING(MAX_NUM_VERTEX)*num_whole_elements+ 
         SIZE_LINE_STRING(rest)*is_there_rest;

    line_string_bound(poly, n, &xmin, &ymin, &xmax, &ymax);

    if(mode == CD_OPEN_LINES)
      startComplexChain(ctxcanvas, (unsigned short) (num_whole_elements+((rest > 0) ? 1 : 0)),
                         size, xmin, ymin, xmax, ymax);
    else
      startComplexShape(ctxcanvas, (unsigned short) (num_whole_elements+((rest > 0) ? 1 : 0)),
                        is_fill, size, xmin, ymin, xmax, ymax);

    for(count=0;count < num_whole_vertex; count+=MAX_NUM_VERTEX, n-=MAX_NUM_VERTEX)
      putLineString(ctxcanvas, &poly[count], MAX_NUM_VERTEX);

    if(rest)
      putLineString(ctxcanvas, &poly[num_whole_vertex],n);

    endComplexElement(ctxcanvas);
  }
  else
  {
    if(is_fill)
      putShape(ctxcanvas, poly, n);
    else
      putLineString(ctxcanvas, poly, n);
  }
}

/**************
 * attributes *
 **************/

static int cdlinestyle (cdCtxCanvas* ctxcanvas, int style)
{
  switch(style)
  {
  case CD_CONTINUOUS:
    ctxcanvas->style = 0;
    break;

  case CD_DASHED:
    ctxcanvas->style = 3;
    break;

  case CD_DOTTED:
    ctxcanvas->style = 1;
    break;

  case CD_DASH_DOT:
    ctxcanvas->style = 4;
    break;

  case CD_DASH_DOT_DOT:
    ctxcanvas->style = 6;
    break;
  }

  return style;
}

static int cdlinewidth (cdCtxCanvas* ctxcanvas, int width)
{
  (void)ctxcanvas;
  width = width & 31;
  return width;
}

static int cdfont(cdCtxCanvas* ctxcanvas, const char *type_face, int style, int size)
{
  (void)style;
  ctxcanvas->tl = (long)(cdGetFontSizePoints(ctxcanvas->canvas, size)/4)*3;

  if (cdStrEqualNoCase(type_face, "Courier"))
    ctxcanvas->typeface_index=1;
  else if (cdStrEqualNoCase(type_face, "Times"))
    ctxcanvas->typeface_index=2;
  else if (cdStrEqualNoCase(type_face, "Helvetica"))
    ctxcanvas->typeface_index=3;
  else if (cdStrEqualNoCase(type_face, "System"))
    ctxcanvas->typeface_index=0;
  else
    return 0;

  return 1;
}

static void cdgetfontdim (cdCtxCanvas* ctxcanvas, int *max_width, int *height, int *ascent, int *descent)
{
  int size_pixel;

  if(max_width)
  {
    int a=0;
    *max_width=0;

    while(fontsizes[ctxcanvas->typeface_index][a])
    {
      if(fontsizes[ctxcanvas->typeface_index][a] > *max_width)
        *max_width = fontsizes[ctxcanvas->typeface_index][a];
      a++;
    }
  }

  size_pixel = cdGetFontSizePixels(ctxcanvas->canvas, ctxcanvas->canvas->font_size);

  if(height)  *height  = (size_pixel*3)/2;
  if(ascent)  *ascent  = size_pixel;
  if(descent) *descent = size_pixel/2;
}

static void cdgettextsize (cdCtxCanvas* ctxcanvas, const char *s, int len, int *width, int *height)
{
 int size_pixel = cdGetFontSizePixels(ctxcanvas->canvas, ctxcanvas->canvas->font_size);
 if(height) *height = size_pixel + get_descent(s, len, size_pixel);
 if(width)  *width  = gettextwidth(ctxcanvas, s, len, size_pixel);
}

static int cdtextalignment (cdCtxCanvas* ctxcanvas, int alignment)
{
  ctxcanvas->is_base = 0;

  /* DGN guarda posicao do texto em relacao ao ponto */
    
  switch(alignment)
  {
  case CD_NORTH:
    ctxcanvas->alignment = 8; /* center-bottom */
    break;

  case CD_SOUTH:
    ctxcanvas->alignment = 6; /* center-top */
    break;

  case CD_EAST:
    ctxcanvas->alignment = 1; /* left-center */
    break;

  case CD_WEST:
    ctxcanvas->alignment = 13; /* right-center */
    break;

  case CD_NORTH_EAST:
    ctxcanvas->alignment = 2; /* left-bottom */
    break;

  case CD_NORTH_WEST:
    ctxcanvas->alignment = 14; /* right-bottom */
    break;

  case CD_SOUTH_EAST:
    ctxcanvas->alignment = 0; /* left-top */
    break;

  case CD_SOUTH_WEST:
    ctxcanvas->alignment = 12; /* right-top */
    break;

  case CD_CENTER:
    ctxcanvas->alignment = 7; /* center-center */
    break;

  case CD_BASE_LEFT:
    ctxcanvas->alignment = 13; /* right-center */
    ctxcanvas->is_base=1;
    break;

  case CD_BASE_CENTER:
    ctxcanvas->alignment = 7; /* center-center */
    ctxcanvas->is_base=1;
    break;

  case CD_BASE_RIGHT:
    ctxcanvas->alignment = 1; /* left-center */
    ctxcanvas->is_base=1;
    break;
  }

  return alignment;
}

/******************************************************/
/* color                                              */
/******************************************************/

static void cdpalette (cdCtxCanvas* ctxcanvas, int n, const long int *palette, int mode)
{
  int c=0;

  IGNORE(mode);
  
  for(c=0; c < n; c++)
    ctxcanvas->colortable[c] = *palette++;

  ctxcanvas->num_colors = n;
}

static long int cdforeground (cdCtxCanvas* ctxcanvas, long int color)
{
  ctxcanvas->color = getclosestColor(ctxcanvas, color);
  return color;
}

/******************************************************/
/* client images                                      */
/******************************************************/

static void cdputimagerectmap(cdCtxCanvas* ctxcanvas, int iw, int ih, const unsigned char *index,
			 const long int *colors, int x, int y, int w, int h, int xmin, int xmax, int ymin, int ymax)
{
  int i=0,j=0, remainder=iw%2, total_colors;
  int *ix=NULL,*iy=NULL;
  Elm_hdr ehdr;
  Disp_hdr dhdr; 
  unsigned char map_colors[256];


  /* raster header element */
  ehdr.type.flags.level=ctxcanvas->level;
  ehdr.type.flags.type=87;
  ehdr.words=39;
  ehdr.xmin=x;
  ehdr.xmax=x+w;
  ehdr.ymin=y;
  ehdr.ymax=y+h;

  putElementHeader(ctxcanvas, &ehdr);

    /* Display Header */
  dhdr.attindx=25;
  dhdr.symb.s=0;

  putDisplayHeader(ctxcanvas, &dhdr);

    /* description of the element */
  put_long(ctxcanvas, 21+(23+w/2+w%2)*h);  /* total length of cell */
  put_word(ctxcanvas, 0x0714);         /* raster flags */
  put_word(ctxcanvas, 0x0100);         /* background e foreground colors
                               (nao usados) */
  put_word(ctxcanvas, w);             /* largura da imagem em pixels */
  put_word(ctxcanvas, h);             /* altura da imagem em pixel */ 
  put_long(ctxcanvas, 0);              /* resevado */
  put_as_double(ctxcanvas, 0);            /* resolution (nao usado) */

  put_word(ctxcanvas, 0x4080);         /* scale */
  put_long(ctxcanvas, 0);
  put_word(ctxcanvas, 0);

  put_long(ctxcanvas, x);              /* origem */ 
  put_long(ctxcanvas, y+h);
  put_long(ctxcanvas, 0);

  put_word(ctxcanvas, 0);              /* no de vertices */ 

  ctxcanvas->is_complex = 1; /* elemento complexo */

  /* obtem o maior indice usado na imagem */

  total_colors = 0;
  for (i = 0; i < iw*ih; i++)
  {        
    if (index[i] > total_colors)
      total_colors = index[i];
  }
  total_colors++;

  /* cria tabela para acelerar match de cor na palette */

  for (i = 0; i < total_colors; i++)
  {        
    map_colors[i] = (unsigned char)getclosestColor(ctxcanvas, colors[i]);
  }

  /** salva dados da imagem **/

  /* calcula stretch */

  ix = cdGetZoomTable(w, xmax-xmin+1, xmin);
  iy = cdGetZoomTable(h, ymax-ymin+1, ymin);
 
  for(i=h-1; i >= 0; i--)
  {
      /* raster header element */
    ehdr.type.flags.level=ctxcanvas->level;
    ehdr.type.flags.type=88;
    ehdr.words=21+w/2+remainder;

    putElementHeader(ctxcanvas, &ehdr);

      /* Display Header */
    dhdr.attindx=7+w/2+remainder;
    dhdr.symb.s=0;
    putDisplayHeader(ctxcanvas, &dhdr);

    put_word(ctxcanvas, 0x0714);   /* raster flags */
    put_word(ctxcanvas, 0x0100);   /* background e foreground 
                           colors (nao usados) */

    put_word(ctxcanvas, 0);    /* x offset da origem */
    put_word(ctxcanvas, i);    /* y offset */
    put_word(ctxcanvas, w);    /* numero de pixels neste elemento */
    
    for(j=0; j < w; j++)   
      put_byte(ctxcanvas, map_colors[index[(iy[i])*iw + ix[j]]]);

    if(remainder) put_byte(ctxcanvas, 0);
  }

  ctxcanvas->is_complex = 0;

  free(ix);  /* libera memoria alocada */
  free(iy);
}

/******************************************************/
/* server images                                      */
/******************************************************/

static void cdpixel (cdCtxCanvas* ctxcanvas, int x, int y, long int color)
{
  long  old_color = cdforeground(ctxcanvas, color);
  int old_linestyle = cdlinestyle(ctxcanvas, CD_CONTINUOUS);
  int old_linewidth = cdlinewidth(ctxcanvas, 1);

  cdline(ctxcanvas, x,y,x,y);

  cdforeground(ctxcanvas, old_color);
  cdlinestyle(ctxcanvas, old_linestyle);
  cdlinewidth(ctxcanvas, old_linewidth);
}

static void cdcreatecanvas(cdCanvas* canvas, void *data)
{
  cdCtxCanvas *ctxcanvas;
  char* strdata = (char*)data;
  char words[4][256];
  char filename[10240] = "";
  char seedfile[10240] = "";
  int count  = 0;
  double res = 0;

  if (!data) return;
    
  /* separa palavras da expressao, que e' na forma
     "filename [mm_wxmm_h] [res] [-f] [-sseedfile]" */

  strdata += cdGetFileName(strdata, filename);
  if (filename[0] == 0)
    return;

  sscanf(strdata, "%s %s %s %s", words[0], words[1], words[2], words[3]);
  
  if(!strlen(filename)) /* se nao pegou filename */
    return;
   
  ctxcanvas = (cdCtxCanvas *) malloc(sizeof(cdCtxCanvas));

  /* tenta criar arquivo DGN */

  if((ctxcanvas->file = fopen (filename, "wb"))==NULL)
  {
    free(ctxcanvas);
    return;
  }

  /* verifica se foi passado tamanho do canvas em mm. Se foi,
     extrai-o da string */

  if(sscanf(words[0], "%lgx%lg",
     &canvas->w_mm, &canvas->h_mm) == 2)
  {
    count++; /* incrementa contador de palavras */

    if(canvas->w_mm == 0 || canvas->h_mm == 0)
    { 
      fclose(ctxcanvas->file);
      free(ctxcanvas);
      return;
    }
  }
  else
    canvas->w_mm = canvas->h_mm = 0;
  
  /* Verifica se foi passada resolucao */

  if(sscanf(words[count], "%lg", &res) == 1)
  {
    count++; /* incrementa contador de palavras */

    if(res <= 0)  /* verifica validade da resolucao */
    { 
      fclose(ctxcanvas->file);
      free(ctxcanvas);
      return;
    }
  }
  else
    res = 3.78;

  /* se tamanho em milimetros nao tiver sido inicializado,
     usa como default o tamanho maximo em pixels para fazer as
     contas
   */

  if (canvas->w_mm == 0 || canvas->h_mm == 0)
  {
    canvas->w = INT_MAX;
    canvas->h = INT_MAX;

    canvas->w_mm = canvas->w / res;
    canvas->h_mm = canvas->h / res;
  }
  else
  {
    canvas->w = (long) (canvas->w_mm * res);
    canvas->h = (long) (canvas->h_mm * res);
  }

  canvas->xres = res;
  canvas->yres = res;
  canvas->bpp = 8;
  
  /* verifica se usuario que alterar metodo de fill */

  if (strcmp(words[count], "-f")==0)
  {
    ctxcanvas->fill_type = CONVEX;
    count++;
  }
  else
    ctxcanvas->fill_type = NORMAL;
  
  /* se tiver passado seedfile como argumento */
  if(sscanf(words[count], "-s%s", seedfile) == 1)
  {
    FILE *seed=NULL;
    char *cd_dir = getenv("CDDIR");
    static char newfilename[512];

    if(cd_dir == NULL)
      cd_dir = ".";

    sprintf(newfilename, "%s/%s", cd_dir, seedfile);
        
    count++;

    /* testa concatenando com variavel de ambiente */

    if((seed = fopen (newfilename, "rb"))==NULL)
    { 
      /* tenta abrir usando string passada pelo usuario
         diretamente */

      if((seed = fopen (seedfile, "rb"))==NULL)
      { 
        fclose(ctxcanvas->file);
        free(ctxcanvas);
        return;
      }
    }

    /* concatena seed */  

    fseek(seed, 0, SEEK_SET);
    fseek(ctxcanvas->file, 0, SEEK_SET);

    ctxcanvas->bytes=0;
    dgn_copy(seed, ctxcanvas);
    fclose(seed);
  }
  
  ctxcanvas->canvas = canvas;
  canvas->ctxcanvas = ctxcanvas;

  /* config */

  ctxcanvas->level = 1;

  /** valores default do contexto sao setados **/

  /* texto */

  ctxcanvas->alignment = 12; 
  ctxcanvas->is_base = 1;
  ctxcanvas->typeface_index = 0;
  ctxcanvas->tl=12;

  /* cores */

  memset(ctxcanvas->colortable, 0, 1024); 
  ctxcanvas->colortable[0] = CD_BLACK;
  ctxcanvas->num_colors = 1;

  /* atributos */

  ctxcanvas->color = 1;
  ctxcanvas->style = 0;

  /* DGN */

  ctxcanvas->is_complex=0;
}

static void cdinittable(cdCanvas* canvas)
{
  canvas->cxFlush = cdflush;
  canvas->cxPixel = cdpixel;
  canvas->cxLine = cdline;
  canvas->cxPoly = cdpoly;
  canvas->cxBox = cdbox;
  canvas->cxArc = cdarc;
  canvas->cxSector = cdsector;
  canvas->cxText = cdtext;
  canvas->cxGetFontDim = cdgetfontdim;
  canvas->cxGetTextSize = cdgettextsize;
  canvas->cxPutImageRectMap = cdputimagerectmap;

  canvas->cxLineStyle = cdlinestyle;
  canvas->cxLineWidth = cdlinewidth;
  canvas->cxFont = cdfont;
  canvas->cxTextAlignment = cdtextalignment;
  canvas->cxPalette = cdpalette;
  canvas->cxForeground = cdforeground;

  canvas->cxKillCanvas = cdkillcanvas;
  canvas->cxDeactivate = cddeactivate;
}

/******************************************************/

static cdContext cdDGNContext =
{
  CD_CAP_ALL & ~(CD_CAP_CLEAR | CD_CAP_PLAY | CD_CAP_PATH | CD_CAP_BEZIER | 
                 CD_CAP_IMAGERGBA | CD_CAP_GETIMAGERGB | 
                 CD_CAP_CLIPAREA | CD_CAP_CLIPPOLY |  CD_CAP_RECT | 
                 CD_CAP_LINECAP | CD_CAP_LINEJOIN | CD_CAP_REGION | CD_CAP_CHORD |
                 CD_CAP_IMAGERGB | CD_CAP_IMAGESRV | 
                 CD_CAP_BACKGROUND | CD_CAP_BACKOPACITY | CD_CAP_WRITEMODE | 
                 CD_CAP_HATCH | CD_CAP_STIPPLE | CD_CAP_PATTERN | 
                 CD_CAP_IMAGERGBA | CD_CAP_GETIMAGERGB | 
                 CD_CAP_FPRIMTIVES  | CD_CAP_TEXTORIENTATION),
  0,
  cdcreatecanvas,
  cdinittable,
  NULL,
  NULL,
};

cdContext* cdContextDGN(void)
{
  return &cdDGNContext;
}