mirror of
git://xwords.git.sourceforge.net/gitroot/xwords/xwords
synced 2025-01-03 23:04:08 +01:00
Attempt to find font that best fits string to encosing rect. Works
but only with a hack: need to understand pango better.
This commit is contained in:
parent
63c0cd8aee
commit
72ee1032c0
1 changed files with 80 additions and 68 deletions
146
linux/gtkdraw.c
146
linux/gtkdraw.c
|
@ -63,13 +63,13 @@ insetRect( XP_Rect* r, short i )
|
||||||
#define GTKMIN_W_HT 12
|
#define GTKMIN_W_HT 12
|
||||||
|
|
||||||
static void
|
static void
|
||||||
eraseRect( GtkDrawCtx* dctx, const XP_Rect* rect )
|
gtkEraseRect( GtkDrawCtx* dctx, const XP_Rect* rect )
|
||||||
{
|
{
|
||||||
gdk_draw_rectangle( DRAW_WHAT(dctx),
|
gdk_draw_rectangle( DRAW_WHAT(dctx),
|
||||||
dctx->drawing_area->style->white_gc,
|
dctx->drawing_area->style->white_gc,
|
||||||
TRUE, rect->left, rect->top,
|
TRUE, rect->left, rect->top,
|
||||||
rect->width, rect->height );
|
rect->width, rect->height );
|
||||||
} /* eraseRect */
|
} /* gtkEraseRect */
|
||||||
|
|
||||||
static void
|
static void
|
||||||
frameRect( GtkDrawCtx* dctx, const XP_Rect* rect )
|
frameRect( GtkDrawCtx* dctx, const XP_Rect* rect )
|
||||||
|
@ -103,7 +103,7 @@ static void
|
||||||
gtk_prim_draw_clearRect( DrawCtx* p_dctx, XP_Rect* rect )
|
gtk_prim_draw_clearRect( DrawCtx* p_dctx, XP_Rect* rect )
|
||||||
{
|
{
|
||||||
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
||||||
eraseRect( dctx, rect );
|
gtkEraseRect( dctx, rect );
|
||||||
} /* gtk_prim_draw_clearRect */
|
} /* gtk_prim_draw_clearRect */
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -164,7 +164,7 @@ layout_for_ht( GtkDrawCtx* dctx, XP_U16 ht )
|
||||||
fps, 0 );
|
fps, 0 );
|
||||||
|
|
||||||
char font[32];
|
char font[32];
|
||||||
snprintf( font, sizeof(font), "helvetica normal %d", ht );
|
snprintf( font, sizeof(font), "helvetica normal %dpx", ht );
|
||||||
|
|
||||||
layout = pango_layout_new( dctx->pangoContext );
|
layout = pango_layout_new( dctx->pangoContext );
|
||||||
fps->fontdesc = pango_font_description_from_string( font );
|
fps->fontdesc = pango_font_description_from_string( font );
|
||||||
|
@ -185,8 +185,8 @@ draw_string_at( GtkDrawCtx* dctx, const char* str, XP_U16 fontHt,
|
||||||
const GdkColor* frground, const GdkColor* bkgrnd )
|
const GdkColor* frground, const GdkColor* bkgrnd )
|
||||||
{
|
{
|
||||||
PangoLayout* layout;
|
PangoLayout* layout;
|
||||||
gint x = where->left;
|
gint xx = where->left;
|
||||||
gint y = where->top;
|
gint yy = where->top;
|
||||||
|
|
||||||
layout = layout_for_ht( dctx, fontHt );
|
layout = layout_for_ht( dctx, fontHt );
|
||||||
|
|
||||||
|
@ -198,12 +198,12 @@ draw_string_at( GtkDrawCtx* dctx, const char* str, XP_U16 fontHt,
|
||||||
|
|
||||||
switch( just ) {
|
switch( just ) {
|
||||||
case XP_GTK_JUST_CENTER:
|
case XP_GTK_JUST_CENTER:
|
||||||
x += (where->width - width) / 2;
|
xx += (where->width - width) / 2;
|
||||||
y += (where->height - height) / 2;
|
yy += (where->height - height) / 2;
|
||||||
break;
|
break;
|
||||||
case XP_GTK_JUST_BOTTOMRIGHT:
|
case XP_GTK_JUST_BOTTOMRIGHT:
|
||||||
x += where->width - width;
|
xx += where->width - width;
|
||||||
y += where->height - height;
|
yy += where->height - height;
|
||||||
break;
|
break;
|
||||||
case XP_GTK_JUST_TOPLEFT:
|
case XP_GTK_JUST_TOPLEFT:
|
||||||
default:
|
default:
|
||||||
|
@ -213,7 +213,7 @@ draw_string_at( GtkDrawCtx* dctx, const char* str, XP_U16 fontHt,
|
||||||
}
|
}
|
||||||
|
|
||||||
gdk_draw_layout_with_colors( DRAW_WHAT(dctx), dctx->drawGC,
|
gdk_draw_layout_with_colors( DRAW_WHAT(dctx), dctx->drawGC,
|
||||||
x, y, layout,
|
xx, yy, layout,
|
||||||
frground, bkgrnd );
|
frground, bkgrnd );
|
||||||
} /* draw_string_at */
|
} /* draw_string_at */
|
||||||
|
|
||||||
|
@ -418,7 +418,7 @@ gtk_draw_drawCell( DrawCtx* p_dctx, const XP_Rect* rect, const XP_UCHAR* letter,
|
||||||
XP_Bool showGrid = dctx->globals->gridOn;
|
XP_Bool showGrid = dctx->globals->gridOn;
|
||||||
XP_Bool highlight = (flags & CELL_HIGHLIGHT) != 0;
|
XP_Bool highlight = (flags & CELL_HIGHLIGHT) != 0;
|
||||||
|
|
||||||
eraseRect( dctx, rect );
|
gtkEraseRect( dctx, rect );
|
||||||
|
|
||||||
insetRect( &rectInset, 1 );
|
insetRect( &rectInset, 1 );
|
||||||
|
|
||||||
|
@ -523,7 +523,7 @@ gtkDrawTileImpl( DrawCtx* p_dctx, const XP_Rect* rect, const XP_UCHAR* textP,
|
||||||
XP_Rect insetR = *rect;
|
XP_Rect insetR = *rect;
|
||||||
|
|
||||||
if ( clearBack ) {
|
if ( clearBack ) {
|
||||||
eraseRect( dctx, &insetR );
|
gtkEraseRect( dctx, &insetR );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( val >= 0 ) {
|
if ( val >= 0 ) {
|
||||||
|
@ -642,7 +642,7 @@ gtk_draw_drawTrayDivider( DrawCtx* p_dctx, const XP_Rect* rect,
|
||||||
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
||||||
XP_Rect r = *rect;
|
XP_Rect r = *rect;
|
||||||
|
|
||||||
eraseRect( dctx, &r );
|
gtkEraseRect( dctx, &r );
|
||||||
|
|
||||||
++r.left;
|
++r.left;
|
||||||
r.width -= selected? 2:1;
|
r.width -= selected? 2:1;
|
||||||
|
@ -687,7 +687,7 @@ gtk_draw_clearRect( DrawCtx* p_dctx, const XP_Rect* rectP )
|
||||||
++rect.width;
|
++rect.width;
|
||||||
++rect.top;
|
++rect.top;
|
||||||
|
|
||||||
eraseRect( dctx, &rect );
|
gtkEraseRect( dctx, &rect );
|
||||||
|
|
||||||
} /* gtk_draw_clearRect */
|
} /* gtk_draw_clearRect */
|
||||||
|
|
||||||
|
@ -714,38 +714,62 @@ gtk_draw_scoreBegin( DrawCtx* p_dctx, const XP_Rect* rect,
|
||||||
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
||||||
|
|
||||||
/* gdk_gc_set_clip_rectangle( dctx->drawGC, (GdkRectangle*)rect ); */
|
/* gdk_gc_set_clip_rectangle( dctx->drawGC, (GdkRectangle*)rect ); */
|
||||||
eraseRect( dctx, rect );
|
gtkEraseRect( dctx, rect );
|
||||||
dctx->topFocus = dfs == DFS_TOP;
|
dctx->topFocus = dfs == DFS_TOP;
|
||||||
} /* gtk_draw_scoreBegin */
|
} /* gtk_draw_scoreBegin */
|
||||||
|
|
||||||
|
static PangoLayout*
|
||||||
|
getLayoutToFitRect( GtkDrawCtx* dctx, const char* str, const XP_Rect* rect )
|
||||||
|
{
|
||||||
|
PangoLayout* layout;
|
||||||
|
float ratio, ratioH;
|
||||||
|
int width, height;
|
||||||
|
XP_U16 len = strlen(str);
|
||||||
|
|
||||||
|
/* First measure it using any font at all */
|
||||||
|
layout = layout_for_ht( dctx, 24 );
|
||||||
|
pango_layout_set_text( layout, str, len );
|
||||||
|
pango_layout_get_pixel_size( layout, &width, &height );
|
||||||
|
|
||||||
|
/* Figure the ratio of is to should-be. The smaller of these is the one
|
||||||
|
we must use. */
|
||||||
|
ratio = (float)rect->width / (float)width;
|
||||||
|
ratioH = (float)rect->height / (float)height;
|
||||||
|
if ( ratioH < ratio ) {
|
||||||
|
ratio = ratioH;
|
||||||
|
}
|
||||||
|
height = 24.0 * ratio;
|
||||||
|
|
||||||
|
layout = layout_for_ht( dctx, height );
|
||||||
|
pango_layout_set_text( layout, str, len );
|
||||||
|
return layout;
|
||||||
|
} /* getLayoutToFitRect */
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gtkDrawDrawRemText( DrawCtx* p_dctx, const XP_Rect* r, XP_S16 nTilesLeft,
|
gtkDrawDrawRemText( DrawCtx* p_dctx, const XP_Rect* rect, XP_S16 nTilesLeft,
|
||||||
XP_U16* widthP, XP_U16* heightP )
|
XP_U16* widthP, XP_U16* heightP )
|
||||||
{
|
{
|
||||||
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
||||||
char buf[10];
|
char buf[10];
|
||||||
gint left = r->left;
|
|
||||||
gint top = r->top;
|
|
||||||
XP_Bool draw = !widthP;
|
|
||||||
PangoLayout* layout = layout_for_ht( dctx, r->height );
|
|
||||||
|
|
||||||
sprintf( buf, "rem:%d", nTilesLeft );
|
snprintf( buf, sizeof(buf), "rem:%d", nTilesLeft );
|
||||||
pango_layout_set_text( layout, buf, strlen(buf) );
|
|
||||||
|
|
||||||
if ( draw ) {
|
if ( !!widthP ) {
|
||||||
gdk_draw_layout_with_colors( DRAW_WHAT(dctx), dctx->drawGC,
|
|
||||||
left, top, layout,
|
|
||||||
&dctx->black, NULL );
|
|
||||||
} else {
|
|
||||||
int width, height;
|
int width, height;
|
||||||
|
PangoLayout* layout = getLayoutToFitRect( dctx, buf, rect );
|
||||||
pango_layout_get_pixel_size( layout, &width, &height );
|
pango_layout_get_pixel_size( layout, &width, &height );
|
||||||
|
if ( width > rect->width ) {
|
||||||
if ( height > GTK_HOR_SCORE_HEIGHT ) {
|
width = rect->width;
|
||||||
height = GTK_HOR_SCORE_HEIGHT;
|
}
|
||||||
|
if ( height > rect->height ) {
|
||||||
|
height = rect->height;
|
||||||
}
|
}
|
||||||
|
|
||||||
*widthP = width;
|
*widthP = width;
|
||||||
*heightP = height;
|
*heightP = height;
|
||||||
|
} else {
|
||||||
|
draw_string_at( dctx, buf, rect->height,
|
||||||
|
rect, XP_GTK_JUST_TOPLEFT,
|
||||||
|
&dctx->black, NULL );
|
||||||
}
|
}
|
||||||
} /* gtkDrawDrawRemText */
|
} /* gtkDrawDrawRemText */
|
||||||
|
|
||||||
|
@ -764,50 +788,43 @@ gtk_draw_drawRemText( DrawCtx* p_dctx, const XP_Rect* rInner,
|
||||||
} /* gtk_draw_drawRemText */
|
} /* gtk_draw_drawRemText */
|
||||||
|
|
||||||
static void
|
static void
|
||||||
scoreWidthAndText( GtkDrawCtx* XP_UNUSED(dctx), PangoLayout* layout, char* buf,
|
formatScoreText( char* buf, XP_U16 bufLen, const DrawScoreInfo* dsi )
|
||||||
const DrawScoreInfo* dsi, XP_U16* widthP, XP_U16* heightP )
|
|
||||||
{
|
{
|
||||||
XP_S16 score = dsi->totalScore;
|
XP_S16 score = dsi->totalScore;
|
||||||
XP_U16 nTilesLeft = dsi->nTilesLeft;
|
XP_U16 nTilesLeft = dsi->nTilesLeft;
|
||||||
XP_Bool isTurn = dsi->isTurn;
|
XP_Bool isTurn = dsi->isTurn;
|
||||||
|
int used;
|
||||||
char* borders = "";
|
char* borders = "";
|
||||||
|
|
||||||
if ( isTurn ) {
|
if ( isTurn ) {
|
||||||
borders = "*";
|
borders = "*";
|
||||||
}
|
}
|
||||||
|
|
||||||
sprintf( buf, "%s%.3d", borders, score );
|
used = snprintf( buf, bufLen, "%s%.3d", borders, score );
|
||||||
if ( (nTilesLeft < MAX_TRAY_TILES) && (nTilesLeft > 0) ) {
|
if ( (nTilesLeft < MAX_TRAY_TILES) && (nTilesLeft > 0) ) {
|
||||||
char nbuf[10];
|
char nbuf[10];
|
||||||
sprintf( nbuf, ":%d", nTilesLeft );
|
sprintf( nbuf, ":%d", nTilesLeft );
|
||||||
(void)strcat( buf, nbuf );
|
(void)strcat( buf, nbuf );
|
||||||
}
|
}
|
||||||
(void)strcat( buf, borders );
|
snprintf( buf+used, bufLen-used, "%s", borders );
|
||||||
|
} /* formatScoreText */
|
||||||
if ( !!widthP || !!heightP ) {
|
|
||||||
int height, width;
|
|
||||||
pango_layout_set_text( layout, buf, strlen(buf) );
|
|
||||||
pango_layout_get_pixel_size( layout, &width, &height );
|
|
||||||
if ( !!widthP ) {
|
|
||||||
*widthP = width;
|
|
||||||
}
|
|
||||||
if ( !!heightP ) {
|
|
||||||
if ( height > GTK_HOR_SCORE_HEIGHT ) {
|
|
||||||
height = GTK_HOR_SCORE_HEIGHT;
|
|
||||||
}
|
|
||||||
*heightP = height;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} /* scoreWidthAndText */
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
gtk_draw_measureScoreText( DrawCtx* p_dctx, const XP_Rect* r,
|
gtk_draw_measureScoreText( DrawCtx* p_dctx, const XP_Rect* bounds,
|
||||||
const DrawScoreInfo* dsi,
|
const DrawScoreInfo* dsi,
|
||||||
XP_U16* width, XP_U16* height )
|
XP_U16* widthP, XP_U16* heightP )
|
||||||
{
|
{
|
||||||
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
||||||
char buf[20];
|
char buf[20];
|
||||||
PangoLayout* layout = layout_for_ht( dctx, r->height-2 );
|
PangoLayout* layout;
|
||||||
scoreWidthAndText( dctx, layout, buf, dsi, width, height );
|
int height, width;
|
||||||
|
|
||||||
|
formatScoreText( buf, sizeof(buf), dsi );
|
||||||
|
layout = getLayoutToFitRect( dctx, buf, bounds );
|
||||||
|
pango_layout_get_pixel_size( layout, &width, &height );
|
||||||
|
|
||||||
|
*widthP = width;
|
||||||
|
*heightP = height;
|
||||||
} /* gtk_draw_measureScoreText */
|
} /* gtk_draw_measureScoreText */
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -816,13 +833,8 @@ gtk_draw_score_drawPlayer( DrawCtx* p_dctx, const XP_Rect* rInner,
|
||||||
{
|
{
|
||||||
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
GtkDrawCtx* dctx = (GtkDrawCtx*)p_dctx;
|
||||||
char scoreBuf[20];
|
char scoreBuf[20];
|
||||||
XP_U16 x;
|
|
||||||
XP_U16 width;
|
|
||||||
|
|
||||||
PangoLayout* layout = layout_for_ht( dctx, rInner->height );
|
formatScoreText( scoreBuf, sizeof(scoreBuf), dsi );
|
||||||
scoreWidthAndText( dctx, layout, scoreBuf, dsi,
|
|
||||||
&width, NULL );
|
|
||||||
x = rInner->left + ((rInner->width - width) /2);
|
|
||||||
|
|
||||||
gdk_gc_set_foreground( dctx->drawGC, &dctx->playerColors[dsi->playerNum] );
|
gdk_gc_set_foreground( dctx->drawGC, &dctx->playerColors[dsi->playerNum] );
|
||||||
|
|
||||||
|
@ -830,7 +842,7 @@ gtk_draw_score_drawPlayer( DrawCtx* p_dctx, const XP_Rect* rInner,
|
||||||
gdk_draw_rectangle( DRAW_WHAT(dctx), dctx->drawGC,
|
gdk_draw_rectangle( DRAW_WHAT(dctx), dctx->drawGC,
|
||||||
TRUE, rOuter->left, rOuter->top,
|
TRUE, rOuter->left, rOuter->top,
|
||||||
rOuter->width, rOuter->height );
|
rOuter->width, rOuter->height );
|
||||||
eraseRect( dctx, rInner );
|
gtkEraseRect( dctx, rInner );
|
||||||
}
|
}
|
||||||
|
|
||||||
draw_string_at( dctx, scoreBuf, rInner->height - 1,
|
draw_string_at( dctx, scoreBuf, rInner->height - 1,
|
||||||
|
@ -862,7 +874,7 @@ gtk_draw_score_pendingScore( DrawCtx* p_dctx, const XP_Rect* rect,
|
||||||
|
|
||||||
localR = *rect;
|
localR = *rect;
|
||||||
insetRect( &localR, 1 );
|
insetRect( &localR, 1 );
|
||||||
eraseRect( dctx, &localR );
|
gtkEraseRect( dctx, &localR );
|
||||||
|
|
||||||
ht = localR.height >> 2;
|
ht = localR.height >> 2;
|
||||||
draw_string_at( dctx, "Pts:", ht,
|
draw_string_at( dctx, "Pts:", ht,
|
||||||
|
@ -903,7 +915,7 @@ gtk_draw_drawTimer( DrawCtx* p_dctx, const XP_Rect* rInner,
|
||||||
gtkFormatTimerText( buf, secondsLeft );
|
gtkFormatTimerText( buf, secondsLeft );
|
||||||
|
|
||||||
/* gdk_gc_set_clip_rectangle( dctx->drawGC, (GdkRectangle*)rInner ); */
|
/* gdk_gc_set_clip_rectangle( dctx->drawGC, (GdkRectangle*)rInner ); */
|
||||||
eraseRect( dctx, rInner );
|
gtkEraseRect( dctx, rInner );
|
||||||
draw_string_at( dctx, buf, rInner->height-1,
|
draw_string_at( dctx, buf, rInner->height-1,
|
||||||
rInner, XP_GTK_JUST_CENTER,
|
rInner, XP_GTK_JUST_CENTER,
|
||||||
&dctx->black, NULL );
|
&dctx->black, NULL );
|
||||||
|
@ -962,7 +974,7 @@ gtk_draw_drawMiniWindow( DrawCtx* p_dctx, const XP_UCHAR* text,
|
||||||
|
|
||||||
/* play some skanky games to get the shadow drawn under and to the
|
/* play some skanky games to get the shadow drawn under and to the
|
||||||
right... */
|
right... */
|
||||||
eraseRect( dctx, &localR );
|
gtkEraseRect( dctx, &localR );
|
||||||
|
|
||||||
insetRect( &localR, 1 );
|
insetRect( &localR, 1 );
|
||||||
--localR.width;
|
--localR.width;
|
||||||
|
@ -971,7 +983,7 @@ gtk_draw_drawMiniWindow( DrawCtx* p_dctx, const XP_UCHAR* text,
|
||||||
|
|
||||||
--localR.top;
|
--localR.top;
|
||||||
--localR.left;
|
--localR.left;
|
||||||
eraseRect( dctx, &localR );
|
gtkEraseRect( dctx, &localR );
|
||||||
frameRect( dctx, &localR );
|
frameRect( dctx, &localR );
|
||||||
|
|
||||||
draw_string_at( dctx, text, GTKMIN_W_HT,
|
draw_string_at( dctx, text, GTKMIN_W_HT,
|
||||||
|
|
Loading…
Reference in a new issue