2003-06-30 16:44:03 +04:00
|
|
|
/*
|
|
|
|
* This file is part of NetSurf, http://netsurf.sourceforge.net/
|
|
|
|
* Licensed under the GNU General Public License,
|
2004-11-03 05:29:01 +03:00
|
|
|
* http://www.opensource.org/licenses/gpl-license
|
2005-02-20 16:19:19 +03:00
|
|
|
* Copyright 2005 James Bursa <bursa@users.sourceforge.net>
|
2002-07-28 01:10:45 +04:00
|
|
|
*/
|
|
|
|
|
2004-01-22 02:57:19 +03:00
|
|
|
/** \file
|
|
|
|
* Font handling (RISC OS implementation).
|
|
|
|
*
|
2005-02-20 16:19:19 +03:00
|
|
|
* The RUfl is used handle and render fonts.
|
2004-01-22 02:57:19 +03:00
|
|
|
*/
|
|
|
|
|
2002-09-27 01:38:33 +04:00
|
|
|
#include <assert.h>
|
2005-02-23 02:36:14 +03:00
|
|
|
#include <string.h>
|
2005-02-20 16:19:19 +03:00
|
|
|
#include "rufl.h"
|
2003-04-04 19:19:32 +04:00
|
|
|
#include "netsurf/css/css.h"
|
2003-07-18 03:01:02 +04:00
|
|
|
#include "netsurf/render/font.h"
|
2003-12-27 03:11:57 +03:00
|
|
|
#include "netsurf/riscos/gui.h"
|
2004-07-06 00:19:52 +04:00
|
|
|
#include "netsurf/riscos/options.h"
|
2002-10-05 21:50:21 +04:00
|
|
|
#include "netsurf/utils/log.h"
|
2005-02-23 02:36:14 +03:00
|
|
|
#include "netsurf/utils/messages.h"
|
|
|
|
#include "netsurf/utils/utils.h"
|
2003-07-18 03:01:02 +04:00
|
|
|
|
2004-09-14 03:56:18 +04:00
|
|
|
|
2005-02-23 02:36:14 +03:00
|
|
|
static void nsfont_check_option(char **option, const char *family,
|
|
|
|
const char *fallback);
|
|
|
|
static bool nsfont_exists(const char *font_family);
|
|
|
|
static int nsfont_list_cmp(const void *keyval, const void *datum);
|
|
|
|
static void nsfont_check_fonts(void);
|
2005-02-20 16:19:19 +03:00
|
|
|
static void nsfont_read_style(const struct css_style *style,
|
|
|
|
const char **font_family, unsigned int *font_size,
|
|
|
|
rufl_style *font_style);
|
2004-01-31 01:28:32 +03:00
|
|
|
|
2002-09-11 18:24:02 +04:00
|
|
|
|
2005-02-23 02:36:14 +03:00
|
|
|
/**
|
|
|
|
* Initialize font handling.
|
|
|
|
*
|
|
|
|
* Exits through die() on error.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void nsfont_init(void)
|
|
|
|
{
|
|
|
|
const char *fallback = "Homerton";
|
|
|
|
rufl_code code;
|
|
|
|
|
|
|
|
nsfont_check_fonts();
|
|
|
|
|
|
|
|
code = rufl_init();
|
|
|
|
if (code != rufl_OK) {
|
|
|
|
if (code == rufl_FONT_MANAGER_ERROR)
|
|
|
|
LOG(("rufl_init: rufl_FONT_MANAGER_ERROR: 0x%x: %s",
|
|
|
|
rufl_fm_error->errnum,
|
|
|
|
rufl_fm_error->errmess));
|
|
|
|
else
|
|
|
|
LOG(("rufl_init: 0x%x", code));
|
|
|
|
die("The Unicode font library could not be initialized. "
|
|
|
|
"Please report this to the developers.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rufl_family_list_entries == 0)
|
|
|
|
die("No fonts could be found. At least one font must be "
|
|
|
|
"installed.");
|
|
|
|
|
|
|
|
if (!nsfont_exists("Homerton"))
|
|
|
|
fallback = rufl_family_list[0];
|
|
|
|
|
|
|
|
nsfont_check_option(&option_font_sans, "Homerton", fallback);
|
|
|
|
nsfont_check_option(&option_font_serif, "Trinity", fallback);
|
|
|
|
nsfont_check_option(&option_font_mono, "Corpus", fallback);
|
|
|
|
nsfont_check_option(&option_font_cursive, "Churchill", fallback);
|
|
|
|
nsfont_check_option(&option_font_fantasy, "Sassoon", fallback);
|
|
|
|
|
|
|
|
if (option_font_default != CSS_FONT_FAMILY_SANS_SERIF &&
|
|
|
|
option_font_default != CSS_FONT_FAMILY_SERIF &&
|
|
|
|
option_font_default != CSS_FONT_FAMILY_MONOSPACE &&
|
|
|
|
option_font_default != CSS_FONT_FAMILY_CURSIVE &&
|
|
|
|
option_font_default != CSS_FONT_FAMILY_FANTASY)
|
|
|
|
option_font_default = CSS_FONT_FAMILY_SANS_SERIF;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check that a font option is valid, and fix it if not.
|
|
|
|
*
|
|
|
|
* \param option pointer to option, as used by options.[ch]
|
|
|
|
* \param family font family to use if option is not set, or the set
|
|
|
|
* family is not available
|
|
|
|
* \param fallback font family to use if family is not available either
|
|
|
|
*/
|
|
|
|
|
|
|
|
void nsfont_check_option(char **option, const char *family,
|
|
|
|
const char *fallback)
|
|
|
|
{
|
|
|
|
if (*option && !nsfont_exists(*option)) {
|
|
|
|
free(*option);
|
|
|
|
*option = 0;
|
|
|
|
}
|
|
|
|
if (!*option) {
|
|
|
|
if (nsfont_exists(family))
|
|
|
|
*option = strdup(family);
|
|
|
|
else
|
|
|
|
*option = strdup(fallback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if a font family is available.
|
|
|
|
*
|
|
|
|
* \param font_family name of font family
|
|
|
|
* \return true if the family is available
|
|
|
|
*/
|
|
|
|
|
|
|
|
bool nsfont_exists(const char *font_family)
|
|
|
|
{
|
|
|
|
if (bsearch(font_family, rufl_family_list,
|
|
|
|
rufl_family_list_entries, sizeof rufl_family_list[0],
|
|
|
|
nsfont_list_cmp))
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int nsfont_list_cmp(const void *keyval, const void *datum)
|
|
|
|
{
|
|
|
|
const char *key = keyval;
|
|
|
|
const char * const *entry = datum;
|
|
|
|
return strcmp(key, *entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check that at least Homerton.Medium is available.
|
|
|
|
*/
|
|
|
|
|
|
|
|
void nsfont_check_fonts(void)
|
|
|
|
{
|
|
|
|
char s[252];
|
|
|
|
font_f font;
|
|
|
|
os_error *error;
|
|
|
|
|
|
|
|
error = xfont_find_font("Homerton.Medium\\ELatin1",
|
|
|
|
160, 160, 0, 0, &font, 0, 0);
|
|
|
|
if (error) {
|
|
|
|
if (error->errnum == error_FILE_NOT_FOUND) {
|
|
|
|
xwimp_start_task("TaskWindow -wimpslot 200K -quit "
|
|
|
|
"<NetSurf$Dir>.FixFonts", 0);
|
|
|
|
die("FontBadInst");
|
|
|
|
} else {
|
|
|
|
LOG(("xfont_find_font: 0x%x: %s",
|
|
|
|
error->errnum, error->errmess));
|
|
|
|
snprintf(s, sizeof s, messages_get("FontError"),
|
|
|
|
error->errmess);
|
|
|
|
die(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
error = xfont_lose_font(font);
|
|
|
|
if (error) {
|
|
|
|
LOG(("xfont_lose_font: 0x%x: %s",
|
|
|
|
error->errnum, error->errmess));
|
|
|
|
snprintf(s, sizeof s, messages_get("FontError"),
|
|
|
|
error->errmess);
|
|
|
|
die(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-27 01:38:33 +04:00
|
|
|
/**
|
2005-02-20 16:19:19 +03:00
|
|
|
* Measure the width of a string.
|
2004-01-22 02:57:19 +03:00
|
|
|
*
|
2005-02-20 16:19:19 +03:00
|
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
|
|
* CSS_FONT_SIZE_LENGTH
|
|
|
|
* \param string UTF-8 string to measure
|
|
|
|
* \param length length of string
|
|
|
|
* \param width updated to width of string[0..length)
|
|
|
|
* \return true on success, false on error and error reported
|
2002-09-27 01:38:33 +04:00
|
|
|
*/
|
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
bool nsfont_width(const struct css_style *style,
|
|
|
|
const char *string, size_t length,
|
|
|
|
int *width)
|
2002-09-27 01:38:33 +04:00
|
|
|
{
|
2005-02-20 16:19:19 +03:00
|
|
|
const char *font_family;
|
|
|
|
unsigned int font_size;
|
|
|
|
rufl_style font_style;
|
|
|
|
rufl_code code;
|
|
|
|
|
|
|
|
nsfont_read_style(style, &font_family, &font_size, &font_style);
|
|
|
|
|
|
|
|
code = rufl_width(font_family, font_style, font_size,
|
|
|
|
string, length,
|
|
|
|
width);
|
|
|
|
if (code != rufl_OK) {
|
|
|
|
if (code == rufl_FONT_MANAGER_ERROR)
|
|
|
|
LOG(("rufl_width: rufl_FONT_MANAGER_ERROR: 0x%x: %s",
|
|
|
|
rufl_fm_error->errnum,
|
|
|
|
rufl_fm_error->errmess));
|
|
|
|
else
|
|
|
|
LOG(("rufl_width: 0x%x", code));
|
|
|
|
/* warn_user("MiscError", "font error"); */
|
|
|
|
return false;
|
2002-09-27 01:38:33 +04:00
|
|
|
}
|
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
*width /= 2;
|
|
|
|
return true;
|
2002-09-27 01:38:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-22 02:57:19 +03:00
|
|
|
/**
|
2005-02-20 16:19:19 +03:00
|
|
|
* Find the position in a string where an x coordinate falls.
|
2004-01-22 02:57:19 +03:00
|
|
|
*
|
2005-02-20 16:19:19 +03:00
|
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
|
|
* CSS_FONT_SIZE_LENGTH
|
|
|
|
* \param string UTF-8 string to measure
|
|
|
|
* \param length length of string
|
|
|
|
* \param x x coordinate to search for
|
|
|
|
* \param char_offset updated to offset in string of actual_x, [0..length]
|
|
|
|
* \param actual_x updated to x coordinate of character closest to x
|
|
|
|
* \return true on success, false on error and error reported
|
2004-01-22 02:57:19 +03:00
|
|
|
*/
|
2004-07-06 00:19:52 +04:00
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
bool nsfont_position_in_string(const struct css_style *style,
|
|
|
|
const char *string, size_t length,
|
|
|
|
int x, size_t *char_offset, int *actual_x)
|
2004-07-06 00:19:52 +04:00
|
|
|
{
|
2005-02-20 16:19:19 +03:00
|
|
|
const char *font_family;
|
|
|
|
unsigned int font_size;
|
|
|
|
rufl_style font_style;
|
|
|
|
rufl_code code;
|
|
|
|
|
|
|
|
nsfont_read_style(style, &font_family, &font_size, &font_style);
|
|
|
|
|
|
|
|
code = rufl_x_to_offset(font_family, font_style, font_size,
|
|
|
|
string, length,
|
|
|
|
x * 2, char_offset, actual_x);
|
|
|
|
if (code != rufl_OK) {
|
|
|
|
if (code == rufl_FONT_MANAGER_ERROR)
|
|
|
|
LOG(("rufl_x_to_offset: rufl_FONT_MANAGER_ERROR: "
|
|
|
|
"0x%x: %s",
|
|
|
|
rufl_fm_error->errnum,
|
|
|
|
rufl_fm_error->errmess));
|
|
|
|
else
|
|
|
|
LOG(("rufl_x_to_offset: 0x%x", code));
|
|
|
|
/* warn_user("MiscError", "font error"); */
|
|
|
|
return false;
|
2004-07-06 00:19:52 +04:00
|
|
|
}
|
2002-09-27 01:38:33 +04:00
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
*actual_x /= 2;
|
|
|
|
return true;
|
2002-09-27 01:38:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-22 02:57:19 +03:00
|
|
|
/**
|
2005-02-20 16:19:19 +03:00
|
|
|
* Find where to split a string to make it fit a width.
|
|
|
|
*
|
|
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
|
|
* CSS_FONT_SIZE_LENGTH
|
|
|
|
* \param string UTF-8 string to measure
|
|
|
|
* \param length length of string
|
|
|
|
* \param x width available
|
|
|
|
* \param char_offset updated to offset in string of actual_x, [0..length]
|
|
|
|
* \param actual_x updated to x coordinate of character closest to x
|
|
|
|
* \return true on success, false on error and error reported
|
2004-01-22 02:57:19 +03:00
|
|
|
*
|
2005-02-20 16:19:19 +03:00
|
|
|
* On exit, [char_offset == 0 ||
|
|
|
|
* string[char_offset] == ' ' ||
|
|
|
|
* char_offset == length]
|
2004-01-22 02:57:19 +03:00
|
|
|
*/
|
2005-01-02 06:58:21 +03:00
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
bool nsfont_split(const struct css_style *style,
|
|
|
|
const char *string, size_t length,
|
|
|
|
int x, size_t *char_offset, int *actual_x)
|
|
|
|
{
|
|
|
|
const char *font_family;
|
|
|
|
unsigned int font_size;
|
|
|
|
rufl_style font_style;
|
|
|
|
rufl_code code;
|
|
|
|
|
|
|
|
nsfont_read_style(style, &font_family, &font_size, &font_style);
|
|
|
|
|
|
|
|
code = rufl_x_to_offset(font_family, font_style, font_size,
|
|
|
|
string, length,
|
|
|
|
x * 2, char_offset, actual_x);
|
|
|
|
if (code != rufl_OK) {
|
|
|
|
if (code == rufl_FONT_MANAGER_ERROR)
|
|
|
|
LOG(("rufl_x_to_offset: rufl_FONT_MANAGER_ERROR: "
|
|
|
|
"0x%x: %s",
|
|
|
|
rufl_fm_error->errnum,
|
|
|
|
rufl_fm_error->errmess));
|
|
|
|
else
|
|
|
|
LOG(("rufl_x_to_offset: 0x%x", code));
|
|
|
|
/* warn_user("MiscError", "font error"); */
|
|
|
|
return false;
|
2004-07-06 00:19:52 +04:00
|
|
|
}
|
2005-02-20 16:19:19 +03:00
|
|
|
|
|
|
|
while (*char_offset && string[*char_offset] != ' ')
|
|
|
|
(*char_offset)--;
|
|
|
|
|
|
|
|
code = rufl_width(font_family, font_style, font_size,
|
|
|
|
string, *char_offset,
|
|
|
|
actual_x);
|
|
|
|
if (code != rufl_OK) {
|
|
|
|
if (code == rufl_FONT_MANAGER_ERROR)
|
|
|
|
LOG(("rufl_width: rufl_FONT_MANAGER_ERROR: 0x%x: %s",
|
|
|
|
rufl_fm_error->errnum,
|
|
|
|
rufl_fm_error->errmess));
|
|
|
|
else
|
|
|
|
LOG(("rufl_width: 0x%x", code));
|
|
|
|
/* warn_user("MiscError", "font error"); */
|
|
|
|
return false;
|
2004-01-22 02:57:19 +03:00
|
|
|
}
|
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
*actual_x /= 2;
|
|
|
|
return true;
|
2004-01-22 02:57:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2005-02-20 16:19:19 +03:00
|
|
|
* Paint a string.
|
2004-01-22 02:57:19 +03:00
|
|
|
*
|
2005-02-20 16:19:19 +03:00
|
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
|
|
* CSS_FONT_SIZE_LENGTH
|
|
|
|
* \param string UTF-8 string to measure
|
|
|
|
* \param length length of string
|
|
|
|
* \param x x coordinate
|
|
|
|
* \param y y coordinate
|
|
|
|
* \param scale scale to apply to font size
|
|
|
|
* \param bg background colour
|
|
|
|
* \param c colour for text
|
|
|
|
* \return true on success, false on error and error reported
|
2004-01-22 02:57:19 +03:00
|
|
|
*/
|
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
bool nsfont_paint(struct css_style *style, const char *string,
|
|
|
|
size_t length, int x, int y, float scale)
|
|
|
|
{
|
|
|
|
const char *font_family;
|
|
|
|
unsigned int font_size;
|
|
|
|
rufl_style font_style;
|
|
|
|
rufl_code code;
|
|
|
|
|
|
|
|
nsfont_read_style(style, &font_family, &font_size, &font_style);
|
|
|
|
|
|
|
|
code = rufl_paint(font_family, font_style, font_size * scale,
|
|
|
|
string, length, x, y);
|
|
|
|
if (code != rufl_OK) {
|
|
|
|
if (code == rufl_FONT_MANAGER_ERROR)
|
|
|
|
LOG(("rufl_width: rufl_FONT_MANAGER_ERROR: 0x%x: %s",
|
|
|
|
rufl_fm_error->errnum,
|
|
|
|
rufl_fm_error->errmess));
|
|
|
|
else
|
|
|
|
LOG(("rufl_width: 0x%x", code));
|
2004-01-22 02:57:19 +03:00
|
|
|
}
|
2005-01-02 06:58:21 +03:00
|
|
|
return true;
|
2002-09-27 01:38:33 +04:00
|
|
|
}
|
|
|
|
|
2002-10-12 17:05:16 +04:00
|
|
|
|
2004-01-22 02:57:19 +03:00
|
|
|
/**
|
2005-02-20 16:19:19 +03:00
|
|
|
* Convert a css_style to a font family, size and rufl_style.
|
2004-01-22 02:57:19 +03:00
|
|
|
*
|
2005-02-20 16:19:19 +03:00
|
|
|
* \param style css_style for this text, with style->font_size.size ==
|
|
|
|
* CSS_FONT_SIZE_LENGTH
|
|
|
|
* \param font_family updated to font family
|
|
|
|
* \param font_size updated to font size
|
|
|
|
* \param font_style updated to font style
|
2004-01-22 02:57:19 +03:00
|
|
|
*/
|
2002-10-12 17:05:16 +04:00
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
void nsfont_read_style(const struct css_style *style,
|
|
|
|
const char **font_family, unsigned int *font_size,
|
|
|
|
rufl_style *font_style)
|
|
|
|
{
|
|
|
|
assert(style->font_size.size == CSS_FONT_SIZE_LENGTH);
|
|
|
|
*font_size = css_len2px(&style->font_size.value.length, style) *
|
|
|
|
72.0 / 90.0 * 16.;
|
|
|
|
if (*font_size < option_font_min_size * 1.6)
|
|
|
|
*font_size = option_font_min_size * 1.6;
|
|
|
|
if (1600 < *font_size)
|
|
|
|
*font_size = 1600;
|
2002-10-12 17:05:16 +04:00
|
|
|
|
2005-02-23 02:36:14 +03:00
|
|
|
switch (style->font_family) {
|
|
|
|
case CSS_FONT_FAMILY_SANS_SERIF:
|
|
|
|
*font_family = option_font_sans;
|
|
|
|
break;
|
|
|
|
case CSS_FONT_FAMILY_SERIF:
|
|
|
|
*font_family = option_font_serif;
|
|
|
|
break;
|
|
|
|
case CSS_FONT_FAMILY_MONOSPACE:
|
|
|
|
*font_family = option_font_mono;
|
|
|
|
break;
|
|
|
|
case CSS_FONT_FAMILY_CURSIVE:
|
|
|
|
*font_family = option_font_cursive;
|
|
|
|
break;
|
|
|
|
case CSS_FONT_FAMILY_FANTASY:
|
|
|
|
*font_family = option_font_fantasy;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*font_family = option_font_sans;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
switch (style->font_style) {
|
2005-02-23 02:36:14 +03:00
|
|
|
case CSS_FONT_STYLE_ITALIC:
|
|
|
|
case CSS_FONT_STYLE_OBLIQUE:
|
|
|
|
*font_style = rufl_SLANTED;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
*font_style = rufl_REGULAR;
|
|
|
|
break;
|
2005-02-20 16:19:19 +03:00
|
|
|
}
|
2005-01-02 06:58:21 +03:00
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
switch (style->font_weight) {
|
2005-02-23 02:36:14 +03:00
|
|
|
case CSS_FONT_WEIGHT_BOLD:
|
|
|
|
case CSS_FONT_WEIGHT_600:
|
|
|
|
case CSS_FONT_WEIGHT_700:
|
|
|
|
case CSS_FONT_WEIGHT_800:
|
|
|
|
case CSS_FONT_WEIGHT_900:
|
|
|
|
*font_style += rufl_BOLD;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2002-10-12 17:05:16 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-22 02:57:19 +03:00
|
|
|
|
2005-01-02 06:58:21 +03:00
|
|
|
|
2004-01-22 02:57:19 +03:00
|
|
|
|
|
|
|
|
2005-02-20 16:19:19 +03:00
|
|
|
void nsfont_txtenum(void *font, const char *text,
|
2004-08-14 16:57:02 +04:00
|
|
|
size_t length,
|
2004-07-06 00:19:52 +04:00
|
|
|
unsigned int *width,
|
|
|
|
const char **rofontname,
|
|
|
|
const char **rotext,
|
2004-08-14 16:57:02 +04:00
|
|
|
size_t *rolength,
|
2005-02-20 16:19:19 +03:00
|
|
|
size_t *consumed) { }
|