/*
|
/*
|
* Copyright (c) 2000 Greg Haerr <greg@censoft.com>
|
* Copyright (c) 2000 Greg Haerr <greg@censoft.com>
|
*
|
*
|
* Nano-X Client utility routines
|
* Nano-X Client utility routines
|
*
|
*
|
* These routines are Gr* functionality without requiring new API entry points.
|
* These routines are Gr* functionality without requiring new API entry points.
|
*/
|
*/
|
#include <stdio.h>
|
#include <stdio.h>
|
#include <stdlib.h>
|
#include <stdlib.h>
|
#define MWINCLUDECOLORS
|
#define MWINCLUDECOLORS
|
#include "nano-X.h"
|
#include "nano-X.h"
|
|
|
/*
|
/*
|
* Create new window with passed style, title and location.
|
* Create new window with passed style, title and location.
|
*/
|
*/
|
GR_WINDOW_ID
|
GR_WINDOW_ID
|
GrNewWindowEx(GR_WM_PROPS props, GR_CHAR *title, GR_WINDOW_ID parent,
|
GrNewWindowEx(GR_WM_PROPS props, GR_CHAR *title, GR_WINDOW_ID parent,
|
GR_COORD x, GR_COORD y, GR_SIZE width, GR_SIZE height,
|
GR_COORD x, GR_COORD y, GR_SIZE width, GR_SIZE height,
|
GR_COLOR background)
|
GR_COLOR background)
|
{
|
{
|
GR_WINDOW_ID wid;
|
GR_WINDOW_ID wid;
|
GR_WM_PROPERTIES wmprops;
|
GR_WM_PROPERTIES wmprops;
|
|
|
/* create window with no borders*/
|
/* create window with no borders*/
|
wid = GrNewWindow(parent, x, y, width, height, 0, background, BLACK);
|
wid = GrNewWindow(parent, x, y, width, height, 0, background, BLACK);
|
if (wid) {
|
if (wid) {
|
/* set properties and title*/
|
/* set properties and title*/
|
wmprops.flags = GR_WM_FLAGS_PROPS | GR_WM_FLAGS_TITLE;
|
wmprops.flags = GR_WM_FLAGS_PROPS | GR_WM_FLAGS_TITLE;
|
wmprops.props = props;
|
wmprops.props = props;
|
wmprops.title = title;
|
wmprops.title = title;
|
GrSetWMProperties(wid, &wmprops);
|
GrSetWMProperties(wid, &wmprops);
|
}
|
}
|
return wid;
|
return wid;
|
}
|
}
|
|
|
/* draw an array of lines*/
|
/* draw an array of lines*/
|
void
|
void
|
GrDrawLines(GR_DRAW_ID w, GR_GC_ID gc, GR_POINT *points, GR_COUNT count)
|
GrDrawLines(GR_DRAW_ID w, GR_GC_ID gc, GR_POINT *points, GR_COUNT count)
|
{
|
{
|
GR_POINT beg, end;
|
GR_POINT beg, end;
|
|
|
if (count < 1)
|
if (count < 1)
|
return;
|
return;
|
if (count == 1) {
|
if (count == 1) {
|
GrPoint(w, gc, points->x, points->y);
|
GrPoint(w, gc, points->x, points->y);
|
return;
|
return;
|
}
|
}
|
|
|
beg = *points++;
|
beg = *points++;
|
while (--count > 0) {
|
while (--count > 0) {
|
end = *points++;
|
end = *points++;
|
GrLine(w, gc, beg.x, beg.y, end.x, end.y);
|
GrLine(w, gc, beg.x, beg.y, end.x, end.y);
|
beg = end;
|
beg = end;
|
}
|
}
|
}
|
}
|
|
|
/*
|
/*
|
* Retrofit routine. Use GrNewCursor and GrSetWindowCursor for new code.
|
* Retrofit routine. Use GrNewCursor and GrSetWindowCursor for new code.
|
*/
|
*/
|
GR_CURSOR_ID
|
GR_CURSOR_ID
|
GrSetCursor(GR_WINDOW_ID wid, GR_SIZE width, GR_SIZE height, GR_COORD hotx,
|
GrSetCursor(GR_WINDOW_ID wid, GR_SIZE width, GR_SIZE height, GR_COORD hotx,
|
GR_COORD hoty, GR_COLOR foreground, GR_COLOR background,
|
GR_COORD hoty, GR_COLOR foreground, GR_COLOR background,
|
GR_BITMAP *fgbitmap, GR_BITMAP *bgbitmap)
|
GR_BITMAP *fgbitmap, GR_BITMAP *bgbitmap)
|
{
|
{
|
GR_CURSOR_ID cid;
|
GR_CURSOR_ID cid;
|
|
|
cid = GrNewCursor(width, height, hotx, hoty, foreground,
|
cid = GrNewCursor(width, height, hotx, hoty, foreground,
|
background, fgbitmap, bgbitmap);
|
background, fgbitmap, bgbitmap);
|
if (cid)
|
if (cid)
|
GrSetWindowCursor(wid, cid);
|
GrSetWindowCursor(wid, cid);
|
return cid;
|
return cid;
|
}
|
}
|
|
|
/* byte-reversing table for reversing X bitmaps*/
|
/* byte-reversing table for reversing X bitmaps*/
|
static unsigned char revbyte[256] = {
|
static unsigned char revbyte[256] = {
|
0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,
|
0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,
|
0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,
|
0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,
|
0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,
|
0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,
|
0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,
|
0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,
|
0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,
|
0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,
|
0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,
|
0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,
|
0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,
|
0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,
|
0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,
|
0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,
|
0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,
|
0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,
|
0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,
|
0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,
|
0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,
|
0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,
|
0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,
|
0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,
|
0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,
|
0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,
|
0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,
|
0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,
|
0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,
|
0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,
|
0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,
|
0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,
|
0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,
|
0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,
|
0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,
|
0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,
|
0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,
|
0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,
|
0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,
|
0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,
|
0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,
|
0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,
|
0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,
|
0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,
|
0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,
|
0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,
|
0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,
|
0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,
|
0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,
|
0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,
|
0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,
|
0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,
|
0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,
|
0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,
|
0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,
|
0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,
|
0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,
|
0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,
|
0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,
|
0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,
|
0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,
|
0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,
|
0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff
|
0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff
|
};
|
};
|
|
|
/*
|
/*
|
* Create a GdBitmap-compatible bitmap (16-bit short array) from data bits
|
* Create a GdBitmap-compatible bitmap (16-bit short array) from data bits
|
* flags specify input format
|
* flags specify input format
|
* caller must free return buffer
|
* caller must free return buffer
|
*
|
*
|
* Currently only works if width/height < bits_width/bits_height
|
* Currently only works if width/height < bits_width/bits_height
|
*/
|
*/
|
GR_BITMAP *
|
GR_BITMAP *
|
GrNewBitmapFromData(GR_SIZE width, GR_SIZE height, GR_SIZE bits_width, GR_SIZE bits_height,
|
GrNewBitmapFromData(GR_SIZE width, GR_SIZE height, GR_SIZE bits_width, GR_SIZE bits_height,
|
void *bits, int flags)
|
void *bits, int flags)
|
{
|
{
|
int x, y;
|
int x, y;
|
int xb;
|
int xb;
|
int brev = flags & GR_BMDATA_BYTEREVERSE;
|
int brev = flags & GR_BMDATA_BYTEREVERSE;
|
int bswap = flags & GR_BMDATA_BYTESWAP;
|
int bswap = flags & GR_BMDATA_BYTESWAP;
|
unsigned char * inbuf = bits;
|
unsigned char * inbuf = bits;
|
unsigned char * p;
|
unsigned char * p;
|
GR_BITMAP * buf;
|
GR_BITMAP * buf;
|
/*int size;*/
|
/*int size;*/
|
|
|
/*
|
/*
|
* bit reverse or byte-swap short words in image
|
* bit reverse or byte-swap short words in image
|
* and pad to 16 bits for GrBitmap()
|
* and pad to 16 bits for GrBitmap()
|
*/
|
*/
|
xb = (width+7)/8; /* FIXME: may not be in packed-8bit format*/
|
xb = (width+7)/8; /* FIXME: may not be in packed-8bit format*/
|
/*size = ((xb+1)&~01) * height;*/
|
/*size = ((xb+1)&~01) * height;*/
|
/*buf = (GR_BITMAP *)malloc(size);*/
|
/*buf = (GR_BITMAP *)malloc(size);*/
|
buf = (GR_BITMAP *)malloc(((xb+1)&~01) * height);
|
buf = (GR_BITMAP *)malloc(((xb+1)&~01) * height);
|
if (!buf)
|
if (!buf)
|
return NULL;
|
return NULL;
|
|
|
p = (unsigned char *)buf;
|
p = (unsigned char *)buf;
|
for(y=0; y<height; ++y) {
|
for(y=0; y<height; ++y) {
|
for(x=0; x<xb; ) {
|
for(x=0; x<xb; ) {
|
if (bswap) {
|
if (bswap) {
|
/* byte-swap short words*/
|
/* byte-swap short words*/
|
unsigned char c = *inbuf++;
|
unsigned char c = *inbuf++;
|
p[1] = brev? revbyte[c]: c;
|
p[1] = brev? revbyte[c]: c;
|
if (x < xb-1) {
|
if (x < xb-1) {
|
c = *inbuf++;
|
c = *inbuf++;
|
p[0] = brev? revbyte[c]: c;
|
p[0] = brev? revbyte[c]: c;
|
} else p[0] = 0;
|
} else p[0] = 0;
|
} else {
|
} else {
|
/* no byte swapping*/
|
/* no byte swapping*/
|
unsigned char c = *inbuf++;
|
unsigned char c = *inbuf++;
|
p[1] = brev? revbyte[c]: c;
|
p[1] = brev? revbyte[c]: c;
|
if (x < xb-1) {
|
if (x < xb-1) {
|
c = *inbuf++;
|
c = *inbuf++;
|
p[0] = brev? revbyte[c]: c;
|
p[0] = brev? revbyte[c]: c;
|
} else p[0] = 0;
|
} else p[0] = 0;
|
}
|
}
|
x += 2;
|
x += 2;
|
p += 2;
|
p += 2;
|
}
|
}
|
inbuf += (bits_width+7)/8 - (width+7)/8; /* FIXME*/
|
inbuf += (bits_width+7)/8 - (width+7)/8; /* FIXME*/
|
|
|
/* pad to 16 bits*/
|
/* pad to 16 bits*/
|
if (xb & 1)
|
if (xb & 1)
|
*p++ = 0;
|
*p++ = 0;
|
}
|
}
|
return buf;
|
return buf;
|
}
|
}
|
|
|
/*
|
/*
|
* Create a new pixmap and initialize from bitmap data and fg/bg colors
|
* Create a new pixmap and initialize from bitmap data and fg/bg colors
|
*/
|
*/
|
GR_WINDOW_ID
|
GR_WINDOW_ID
|
GrNewPixmapFromData(GR_SIZE width, GR_SIZE height, GR_COLOR foreground,
|
GrNewPixmapFromData(GR_SIZE width, GR_SIZE height, GR_COLOR foreground,
|
GR_COLOR background, void * bits, int flags)
|
GR_COLOR background, void * bits, int flags)
|
{
|
{
|
GR_WINDOW_ID pid;
|
GR_WINDOW_ID pid;
|
GR_GC_ID gc;
|
GR_GC_ID gc;
|
GR_BITMAP * buf;
|
GR_BITMAP * buf;
|
|
|
pid = GrNewPixmap(width, height, NULL);
|
pid = GrNewPixmap(width, height, NULL);
|
if (pid) {
|
if (pid) {
|
gc = GrNewGC();
|
gc = GrNewGC();
|
GrSetGCForeground(gc, foreground);
|
GrSetGCForeground(gc, foreground);
|
GrSetGCBackground(gc, background);
|
GrSetGCBackground(gc, background);
|
|
|
buf = GrNewBitmapFromData(width, height, width, height, bits, flags);
|
buf = GrNewBitmapFromData(width, height, width, height, bits, flags);
|
if (buf) {
|
if (buf) {
|
GrBitmap(pid, gc, 0, 0, width, height, buf);
|
GrBitmap(pid, gc, 0, 0, width, height, buf);
|
free(buf);
|
free(buf);
|
}
|
}
|
GrDestroyGC(gc);
|
GrDestroyGC(gc);
|
}
|
}
|
return pid;
|
return pid;
|
}
|
}
|
|
|