2008-08-02 18:31:32 +04:00
|
|
|
/*
|
2011-04-22 19:30:16 +04:00
|
|
|
* Copyright 2008 - 2011 Chris Young <chris@unsatisfactorysoftware.co.uk>
|
2008-08-02 18:31:32 +04:00
|
|
|
*
|
|
|
|
* This file is part of NetSurf, http://www.netsurf-browser.org/
|
|
|
|
*
|
|
|
|
* NetSurf is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; version 2 of the License.
|
|
|
|
*
|
|
|
|
* NetSurf is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <assert.h>
|
2011-04-22 20:48:29 +04:00
|
|
|
|
2008-08-10 13:57:41 +04:00
|
|
|
#include "amiga/font.h"
|
2011-04-22 20:48:29 +04:00
|
|
|
#include "amiga/gui.h"
|
2008-08-26 05:21:22 +04:00
|
|
|
#include "amiga/utf8.h"
|
2011-04-30 23:05:11 +04:00
|
|
|
#include "amiga/object.h"
|
2011-04-22 20:48:29 +04:00
|
|
|
#include "amiga/options.h"
|
|
|
|
#include "css/css.h"
|
|
|
|
#include "css/utils.h"
|
|
|
|
#include "render/font.h"
|
|
|
|
#include "utils/log.h"
|
2011-05-02 14:33:16 +04:00
|
|
|
#include "utils/schedule.h"
|
2008-08-30 20:55:25 +04:00
|
|
|
#include "utils/utf8.h"
|
2011-04-22 20:48:29 +04:00
|
|
|
#include "utils/utils.h"
|
|
|
|
|
|
|
|
#include <proto/diskfont.h>
|
|
|
|
#include <proto/exec.h>
|
|
|
|
#include <proto/graphics.h>
|
|
|
|
#include <proto/Picasso96API.h>
|
2011-05-01 15:59:08 +04:00
|
|
|
#include <proto/timer.h>
|
2011-04-22 20:48:29 +04:00
|
|
|
#include <proto/utility.h>
|
|
|
|
|
2008-08-30 20:55:25 +04:00
|
|
|
#include <diskfont/diskfonttag.h>
|
|
|
|
#include <diskfont/oterrors.h>
|
2011-04-22 20:48:29 +04:00
|
|
|
#include <graphics/rpattr.h>
|
|
|
|
|
2011-01-07 02:17:17 +03:00
|
|
|
#ifdef __amigaos4__
|
2008-12-28 02:25:42 +03:00
|
|
|
#include <graphics/blitattr.h>
|
2011-01-07 02:17:17 +03:00
|
|
|
#endif
|
2009-01-16 23:05:21 +03:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
#define NSA_UNICODE_FONT PLOT_FONT_FAMILY_COUNT
|
2008-09-15 22:56:11 +04:00
|
|
|
|
2011-04-22 20:06:46 +04:00
|
|
|
#define NSA_NORMAL 0
|
|
|
|
#define NSA_ITALIC 1
|
|
|
|
#define NSA_BOLD 2
|
|
|
|
#define NSA_BOLDITALIC 3
|
2011-12-04 05:45:31 +04:00
|
|
|
#define NSA_OBLIQUE 4
|
|
|
|
#define NSA_BOLDOBLIQUE 6
|
2011-04-22 20:06:46 +04:00
|
|
|
|
2011-04-22 19:30:16 +04:00
|
|
|
#define NSA_VALUE_BOLDX (1 << 12)
|
2011-05-02 14:33:16 +04:00
|
|
|
#define NSA_VALUE_BOLDY 0
|
2011-04-22 19:30:16 +04:00
|
|
|
#define NSA_VALUE_SHEARSIN (1 << 14)
|
|
|
|
#define NSA_VALUE_SHEARCOS (1 << 16)
|
|
|
|
|
2011-05-07 02:42:37 +04:00
|
|
|
#define NSA_FONT_EMWIDTH(s) (s / FONT_SIZE_SCALE) * (ami_xdpi / 72.0)
|
2011-05-02 14:33:16 +04:00
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
struct ami_font_node
|
|
|
|
{
|
|
|
|
struct OutlineFont *font;
|
|
|
|
char *bold;
|
|
|
|
char *italic;
|
|
|
|
char *bolditalic;
|
2011-05-01 15:59:08 +04:00
|
|
|
struct TimeVal lastused;
|
2011-04-30 23:05:11 +04:00
|
|
|
};
|
2010-05-06 14:02:58 +04:00
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
struct MinList *ami_font_list = NULL;
|
2011-12-04 05:10:19 +04:00
|
|
|
struct List ami_diskfontlib_list;
|
2011-04-22 20:06:46 +04:00
|
|
|
ULONG ami_devicedpi;
|
2011-05-02 14:33:16 +04:00
|
|
|
ULONG ami_xdpi;
|
2011-04-22 20:06:46 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
int32 ami_font_plot_glyph(struct OutlineFont *ofont, struct RastPort *rp,
|
|
|
|
uint16 char1, uint16 char2, uint32 x, uint32 y, uint32 emwidth);
|
2011-05-01 15:59:08 +04:00
|
|
|
struct OutlineFont *ami_open_outline_font(const plot_font_style_t *fstyle,
|
|
|
|
BOOL fallback);
|
|
|
|
static void ami_font_cleanup(struct MinList *ami_font_list);
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
static bool nsfont_width(const plot_font_style_t *fstyle,
|
2008-08-02 18:31:32 +04:00
|
|
|
const char *string, size_t length,
|
|
|
|
int *width);
|
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
static bool nsfont_position_in_string(const plot_font_style_t *fstyle,
|
2008-08-02 18:31:32 +04:00
|
|
|
const char *string, size_t length,
|
|
|
|
int x, size_t *char_offset, int *actual_x);
|
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
static bool nsfont_split(const plot_font_style_t *fstyle,
|
2008-08-02 18:31:32 +04:00
|
|
|
const char *string, size_t length,
|
|
|
|
int x, size_t *char_offset, int *actual_x);
|
|
|
|
|
|
|
|
const struct font_functions nsfont = {
|
|
|
|
nsfont_width,
|
|
|
|
nsfont_position_in_string,
|
|
|
|
nsfont_split
|
|
|
|
};
|
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
bool nsfont_width(const plot_font_style_t *fstyle,
|
2008-08-02 18:31:32 +04:00
|
|
|
const char *string, size_t length,
|
|
|
|
int *width)
|
|
|
|
{
|
2009-07-21 14:59:53 +04:00
|
|
|
*width = ami_unicode_text(NULL,string,length,fstyle,0,0);
|
2009-01-16 23:05:21 +03:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
if(*width <= 0) *width == length; // fudge
|
|
|
|
|
2008-08-02 18:31:32 +04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-08-07 22:44:28 +04:00
|
|
|
/**
|
|
|
|
* Find the position in a string where an x coordinate falls.
|
|
|
|
*
|
2009-07-21 14:59:53 +04:00
|
|
|
* \param fstyle style for this text
|
2008-08-07 22:44:28 +04:00
|
|
|
* \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
|
|
|
|
*/
|
2008-08-02 18:31:32 +04:00
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
bool nsfont_position_in_string(const plot_font_style_t *fstyle,
|
2008-08-02 18:31:32 +04:00
|
|
|
const char *string, size_t length,
|
|
|
|
int x, size_t *char_offset, int *actual_x)
|
|
|
|
{
|
2008-08-10 00:28:35 +04:00
|
|
|
struct TextExtent extent;
|
2009-01-17 03:43:51 +03:00
|
|
|
struct TextFont *tfont;
|
2009-04-01 21:36:18 +04:00
|
|
|
uint16 *utf16 = NULL, *outf16 = NULL;
|
2011-05-02 14:33:16 +04:00
|
|
|
uint16 utf16next = 0;
|
2010-04-30 02:30:21 +04:00
|
|
|
FIXED kern = 0;
|
2010-05-06 14:02:58 +04:00
|
|
|
struct OutlineFont *ofont, *ufont = NULL;
|
2009-04-01 21:36:18 +04:00
|
|
|
struct GlyphMap *glyph;
|
|
|
|
uint32 tx=0,i=0;
|
2009-09-24 01:52:36 +04:00
|
|
|
size_t len, utf8len = 0;
|
2009-04-01 21:36:18 +04:00
|
|
|
uint8 *utf8;
|
|
|
|
uint32 co = 0;
|
2009-09-05 15:33:05 +04:00
|
|
|
int utf16charlen;
|
2011-05-07 02:42:37 +04:00
|
|
|
ULONG emwidth = (ULONG)NSA_FONT_EMWIDTH(fstyle->size);
|
2010-05-06 14:02:58 +04:00
|
|
|
int32 tempx;
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
len = utf8_bounded_length(string, length);
|
2009-04-18 20:55:59 +04:00
|
|
|
if(utf8_to_enc(string,"UTF-16",length,(char **)&utf16) != UTF8_CONVERT_OK) return false;
|
2009-04-01 21:36:18 +04:00
|
|
|
outf16 = utf16;
|
2008-08-22 01:42:48 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
if(!(ofont = ami_open_outline_font(fstyle, FALSE))) return false;
|
2008-09-16 02:18:33 +04:00
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
*char_offset = length;
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2010-05-04 00:57:32 +04:00
|
|
|
for(i=0;i<len;i++)
|
2009-01-17 03:43:51 +03:00
|
|
|
{
|
2009-09-05 15:33:05 +04:00
|
|
|
if (*utf16 < 0xD800 || 0xDFFF < *utf16)
|
|
|
|
utf16charlen = 1;
|
|
|
|
else
|
|
|
|
utf16charlen = 2;
|
|
|
|
|
2010-03-15 01:48:25 +03:00
|
|
|
utf8len = utf8_char_byte_length(string);
|
|
|
|
|
2010-04-30 03:05:52 +04:00
|
|
|
utf16next = utf16[utf16charlen];
|
2010-04-30 02:30:21 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
tempx = ami_font_plot_glyph(ofont, NULL, *utf16, utf16next,
|
|
|
|
0, 0, emwidth);
|
|
|
|
|
|
|
|
if(tempx == 0)
|
2009-01-17 03:43:51 +03:00
|
|
|
{
|
2010-05-06 14:02:58 +04:00
|
|
|
if(ufont == NULL)
|
2009-01-17 03:43:51 +03:00
|
|
|
{
|
2010-05-06 14:02:58 +04:00
|
|
|
ufont = ami_open_outline_font(fstyle, TRUE);
|
|
|
|
}
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
if(ufont)
|
|
|
|
{
|
|
|
|
tempx = ami_font_plot_glyph(ufont, NULL, *utf16, utf16next,
|
|
|
|
0, 0, emwidth);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
if(tempx == 0)
|
|
|
|
{
|
|
|
|
tempx = ami_font_plot_glyph(ofont, NULL, 0xfffd, utf16next,
|
|
|
|
0, 0, emwidth);
|
2009-01-17 03:43:51 +03:00
|
|
|
}
|
2010-05-06 14:02:58 +04:00
|
|
|
*/
|
2009-01-17 03:43:51 +03:00
|
|
|
}
|
2009-09-24 01:52:36 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
if(x < (tx + tempx))
|
|
|
|
{
|
|
|
|
*actual_x = tx;
|
|
|
|
i = len+1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
co += utf8len;
|
|
|
|
}
|
|
|
|
|
|
|
|
tx += tempx;
|
2009-09-24 01:52:36 +04:00
|
|
|
string += utf8len;
|
2009-09-05 15:33:05 +04:00
|
|
|
utf16 += utf16charlen;
|
2009-01-17 03:43:51 +03:00
|
|
|
}
|
2009-09-05 18:39:38 +04:00
|
|
|
|
2010-04-30 21:28:07 +04:00
|
|
|
if(co >= (length))
|
2009-09-05 18:39:38 +04:00
|
|
|
{
|
|
|
|
*actual_x = tx;
|
|
|
|
co = length;
|
|
|
|
}
|
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
*char_offset = co;
|
2009-09-05 18:39:38 +04:00
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
free(outf16);
|
2008-08-02 18:31:32 +04:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-07 22:44:28 +04:00
|
|
|
/**
|
|
|
|
* Find where to split a string to make it fit a width.
|
|
|
|
*
|
2009-07-21 14:59:53 +04:00
|
|
|
* \param fstyle style for this text
|
2008-08-07 22:44:28 +04:00
|
|
|
* \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
|
|
|
|
*
|
|
|
|
* On exit, [char_offset == 0 ||
|
|
|
|
* string[char_offset] == ' ' ||
|
|
|
|
* char_offset == length]
|
|
|
|
*/
|
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
bool nsfont_split(const plot_font_style_t *fstyle,
|
2008-08-02 18:31:32 +04:00
|
|
|
const char *string, size_t length,
|
|
|
|
int x, size_t *char_offset, int *actual_x)
|
|
|
|
{
|
2008-08-10 00:28:35 +04:00
|
|
|
struct TextExtent extent;
|
|
|
|
ULONG co;
|
2011-05-02 14:33:16 +04:00
|
|
|
char *ostr = (char *)string;
|
2009-01-17 03:43:51 +03:00
|
|
|
struct TextFont *tfont;
|
2009-04-01 21:36:18 +04:00
|
|
|
uint16 *utf16 = NULL,*outf16 = NULL;
|
2010-04-30 02:30:21 +04:00
|
|
|
uint16 utf16next = 0;
|
|
|
|
FIXED kern = 0;
|
|
|
|
int utf16charlen = 0;
|
2010-05-06 14:02:58 +04:00
|
|
|
struct OutlineFont *ofont, *ufont = NULL;
|
2009-04-01 21:36:18 +04:00
|
|
|
struct GlyphMap *glyph;
|
|
|
|
uint32 tx=0,i=0;
|
|
|
|
size_t len;
|
2009-09-05 15:33:05 +04:00
|
|
|
int utf8len, utf8clen = 0;
|
2010-05-06 14:02:58 +04:00
|
|
|
int32 tempx = 0;
|
2011-05-07 02:42:37 +04:00
|
|
|
ULONG emwidth = (ULONG)NSA_FONT_EMWIDTH(fstyle->size);
|
2008-08-10 00:28:35 +04:00
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
len = utf8_bounded_length(string, length);
|
2009-04-18 20:55:59 +04:00
|
|
|
if(utf8_to_enc((char *)string,"UTF-16",length,(char **)&utf16) != UTF8_CONVERT_OK) return false;
|
2009-04-01 21:36:18 +04:00
|
|
|
outf16 = utf16;
|
2010-05-06 14:02:58 +04:00
|
|
|
if(!(ofont = ami_open_outline_font(fstyle, FALSE))) return false;
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
*char_offset = 0;
|
2009-09-05 15:58:13 +04:00
|
|
|
*actual_x = 0;
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2010-05-04 00:57:32 +04:00
|
|
|
for(i=0;i<len;i++)
|
2008-08-26 22:03:07 +04:00
|
|
|
{
|
2009-09-05 16:35:05 +04:00
|
|
|
utf8len = utf8_char_byte_length(string+utf8clen);
|
|
|
|
|
2010-04-30 02:30:21 +04:00
|
|
|
if (*utf16 < 0xD800 || 0xDFFF < *utf16)
|
|
|
|
utf16charlen = 1;
|
|
|
|
else
|
|
|
|
utf16charlen = 2;
|
|
|
|
|
2010-04-30 03:05:52 +04:00
|
|
|
utf16next = utf16[utf16charlen];
|
2010-04-30 02:30:21 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
if(x < tx)
|
|
|
|
{
|
|
|
|
i = length+1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if(string[utf8clen] == ' ') //*utf16 == 0x0020)
|
|
|
|
{
|
|
|
|
*actual_x = tx;
|
|
|
|
*char_offset = utf8clen;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tempx = ami_font_plot_glyph(ofont, NULL, *utf16, utf16next, 0, 0, emwidth);
|
|
|
|
|
|
|
|
if(tempx == 0)
|
2009-01-17 03:43:51 +03:00
|
|
|
{
|
2010-05-06 14:02:58 +04:00
|
|
|
if(ufont == NULL)
|
|
|
|
{
|
|
|
|
ufont = ami_open_outline_font(fstyle, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ufont)
|
2009-01-17 03:43:51 +03:00
|
|
|
{
|
2010-05-06 14:02:58 +04:00
|
|
|
tempx = ami_font_plot_glyph(ufont, NULL, *utf16, utf16next,
|
|
|
|
0, 0, emwidth);
|
2009-04-01 21:36:18 +04:00
|
|
|
}
|
2010-05-06 14:02:58 +04:00
|
|
|
/*
|
|
|
|
if(tempx == 0)
|
|
|
|
{
|
|
|
|
tempx = ami_font_plot_glyph(ofont, NULL, 0xfffd, utf16next,
|
|
|
|
0, 0, emwidth);
|
|
|
|
}
|
|
|
|
*/
|
2009-04-01 21:36:18 +04:00
|
|
|
}
|
2009-09-05 15:33:05 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
tx += tempx;
|
2010-04-30 02:30:21 +04:00
|
|
|
utf16 += utf16charlen;
|
2009-09-05 15:33:05 +04:00
|
|
|
utf8clen += utf8len;
|
2008-09-15 22:56:11 +04:00
|
|
|
}
|
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
free(outf16);
|
2008-08-10 13:57:41 +04:00
|
|
|
|
2009-04-01 21:36:18 +04:00
|
|
|
return true;
|
2008-08-17 20:22:40 +04:00
|
|
|
}
|
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
/**
|
|
|
|
* Search for a font in the list and load from disk if not present
|
|
|
|
*/
|
|
|
|
struct ami_font_node *ami_font_open(const char *font)
|
|
|
|
{
|
|
|
|
struct nsObject *node;
|
|
|
|
struct ami_font_node *nodedata;
|
|
|
|
|
|
|
|
node = (struct nsObject *)FindIName((struct List *)ami_font_list, font);
|
2011-05-01 15:59:08 +04:00
|
|
|
if(node)
|
|
|
|
{
|
|
|
|
nodedata = node->objstruct;
|
|
|
|
GetSysTime(&nodedata->lastused);
|
|
|
|
return nodedata;
|
|
|
|
}
|
2011-04-30 23:05:11 +04:00
|
|
|
|
|
|
|
LOG(("Font cache miss: %s", font));
|
|
|
|
|
|
|
|
nodedata = AllocVec(sizeof(struct ami_font_node), MEMF_PRIVATE | MEMF_CLEAR);
|
|
|
|
|
|
|
|
node = AddObject(ami_font_list, AMINS_FONT);
|
|
|
|
if(!node) return NULL;
|
|
|
|
|
|
|
|
node->objstruct = nodedata;
|
|
|
|
node->dtz_Node.ln_Name = strdup(font);
|
|
|
|
|
2011-12-04 05:10:19 +04:00
|
|
|
nodedata->font = OpenOutlineFont(font, &ami_diskfontlib_list, OFF_OPEN);
|
2011-04-30 23:05:11 +04:00
|
|
|
if(!nodedata->font)
|
|
|
|
{
|
|
|
|
LOG(("Requested font not found: %s", font));
|
|
|
|
warn_user("CompError", font);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
nodedata->bold = (char *)GetTagData(OT_BName, 0, nodedata->font->olf_OTagList);
|
2011-05-01 12:53:51 +04:00
|
|
|
if(nodedata->bold)
|
|
|
|
LOG(("Bold font defined for %s is %s", font, nodedata->bold));
|
|
|
|
else
|
|
|
|
LOG(("Warning: No designed bold font defined for %s", font));
|
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
nodedata->italic = (char *)GetTagData(OT_IName, 0, nodedata->font->olf_OTagList);
|
2011-05-01 12:53:51 +04:00
|
|
|
if(nodedata->italic)
|
|
|
|
LOG(("Italic font defined for %s is %s", font, nodedata->italic));
|
|
|
|
else
|
|
|
|
LOG(("Warning: No designed italic font defined for %s", font));
|
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
nodedata->bolditalic = (char *)GetTagData(OT_BIName, 0, nodedata->font->olf_OTagList);
|
2011-05-01 12:53:51 +04:00
|
|
|
if(nodedata->bolditalic)
|
|
|
|
LOG(("Bold-italic font defined for %s is %s", font, nodedata->bolditalic));
|
|
|
|
else
|
|
|
|
LOG(("Warning: No designed bold-italic font defined for %s", font));
|
2011-04-30 23:05:11 +04:00
|
|
|
|
2011-05-01 15:59:08 +04:00
|
|
|
GetSysTime(&nodedata->lastused);
|
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
return nodedata;
|
|
|
|
}
|
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
/**
|
|
|
|
* Open an outline font in the specified size and style
|
|
|
|
*
|
|
|
|
* \param fstyle font style structure
|
|
|
|
* \param default open a default font instead of the one specified by fstyle
|
|
|
|
* \return outline font or NULL on error
|
|
|
|
*/
|
|
|
|
struct OutlineFont *ami_open_outline_font(const plot_font_style_t *fstyle, BOOL fallback)
|
2008-08-30 20:55:25 +04:00
|
|
|
{
|
2011-04-30 23:05:11 +04:00
|
|
|
struct ami_font_node *node;
|
2008-08-30 20:55:25 +04:00
|
|
|
struct OutlineFont *ofont;
|
|
|
|
char *fontname;
|
2009-07-24 03:05:34 +04:00
|
|
|
ULONG ysize;
|
2009-01-20 19:53:06 +03:00
|
|
|
int tstyle = 0;
|
2010-05-06 14:02:58 +04:00
|
|
|
plot_font_generic_family_t fontfamily;
|
2011-04-22 19:30:16 +04:00
|
|
|
ULONG emboldenx = 0;
|
|
|
|
ULONG emboldeny = 0;
|
|
|
|
ULONG shearsin = 0;
|
|
|
|
ULONG shearcos = (1 << 16);
|
2010-05-06 14:02:58 +04:00
|
|
|
|
|
|
|
if(fallback) fontfamily = NSA_UNICODE_FONT;
|
|
|
|
else fontfamily = fstyle->family;
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
switch(fontfamily)
|
|
|
|
{
|
|
|
|
case PLOT_FONT_FAMILY_SANS_SERIF:
|
|
|
|
fontname = option_font_sans;
|
|
|
|
break;
|
|
|
|
case PLOT_FONT_FAMILY_SERIF:
|
|
|
|
fontname = option_font_serif;
|
|
|
|
break;
|
|
|
|
case PLOT_FONT_FAMILY_MONOSPACE:
|
|
|
|
fontname = option_font_mono;
|
|
|
|
break;
|
|
|
|
case PLOT_FONT_FAMILY_CURSIVE:
|
|
|
|
fontname = option_font_cursive;
|
|
|
|
break;
|
|
|
|
case PLOT_FONT_FAMILY_FANTASY:
|
|
|
|
fontname = option_font_fantasy;
|
|
|
|
break;
|
|
|
|
case NSA_UNICODE_FONT:
|
|
|
|
default:
|
|
|
|
fontname = option_font_unicode;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
node = ami_font_open(fontname);
|
|
|
|
if(!node) return NULL;
|
|
|
|
|
2011-12-04 05:45:31 +04:00
|
|
|
if (fstyle->flags & FONTF_OBLIQUE)
|
|
|
|
tstyle = NSA_OBLIQUE;
|
|
|
|
|
|
|
|
if (fstyle->flags & FONTF_ITALIC)
|
|
|
|
tstyle = NSA_ITALIC;
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
if (fstyle->weight >= 700)
|
|
|
|
tstyle += NSA_BOLD;
|
2009-01-20 19:53:06 +03:00
|
|
|
|
|
|
|
switch(tstyle)
|
|
|
|
{
|
|
|
|
case NSA_ITALIC:
|
2011-04-30 23:05:11 +04:00
|
|
|
if(node->italic)
|
2011-04-22 19:30:16 +04:00
|
|
|
{
|
2011-04-30 23:05:11 +04:00
|
|
|
node = ami_font_open(node->italic);
|
2011-05-01 12:49:00 +04:00
|
|
|
if(!node) return NULL;
|
2011-04-22 19:30:16 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
shearsin = NSA_VALUE_SHEARSIN;
|
|
|
|
shearcos = NSA_VALUE_SHEARCOS;
|
|
|
|
}
|
2009-01-20 19:53:06 +03:00
|
|
|
break;
|
|
|
|
|
2011-12-04 05:45:31 +04:00
|
|
|
case NSA_OBLIQUE:
|
|
|
|
shearsin = NSA_VALUE_SHEARSIN;
|
|
|
|
shearcos = NSA_VALUE_SHEARCOS;
|
|
|
|
break;
|
|
|
|
|
2009-01-20 19:53:06 +03:00
|
|
|
case NSA_BOLD:
|
2011-04-30 23:05:11 +04:00
|
|
|
if(node->bold)
|
2011-04-22 18:53:56 +04:00
|
|
|
{
|
2011-04-30 23:05:11 +04:00
|
|
|
node = ami_font_open(node->bold);
|
2011-05-01 12:49:00 +04:00
|
|
|
if(!node) return NULL;
|
2011-04-22 18:53:56 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-04-22 19:30:16 +04:00
|
|
|
emboldenx = NSA_VALUE_BOLDX;
|
|
|
|
emboldeny = NSA_VALUE_BOLDY;
|
2011-04-22 18:53:56 +04:00
|
|
|
}
|
2009-01-20 19:53:06 +03:00
|
|
|
break;
|
|
|
|
|
2011-12-04 05:45:31 +04:00
|
|
|
case NSA_BOLDOBLIQUE:
|
|
|
|
shearsin = NSA_VALUE_SHEARSIN;
|
|
|
|
shearcos = NSA_VALUE_SHEARCOS;
|
|
|
|
|
|
|
|
if(node->bold)
|
|
|
|
{
|
|
|
|
node = ami_font_open(node->bold);
|
|
|
|
if(!node) return NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
emboldenx = NSA_VALUE_BOLDX;
|
|
|
|
emboldeny = NSA_VALUE_BOLDY;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2009-01-20 19:53:06 +03:00
|
|
|
case NSA_BOLDITALIC:
|
2011-04-30 23:05:11 +04:00
|
|
|
if(node->bolditalic)
|
2011-04-22 18:53:56 +04:00
|
|
|
{
|
2011-04-30 23:05:11 +04:00
|
|
|
node = ami_font_open(node->bolditalic);
|
2011-05-01 12:49:00 +04:00
|
|
|
if(!node) return NULL;
|
2011-04-22 18:53:56 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-04-22 19:30:16 +04:00
|
|
|
emboldenx = NSA_VALUE_BOLDX;
|
|
|
|
emboldeny = NSA_VALUE_BOLDY;
|
|
|
|
shearsin = NSA_VALUE_SHEARSIN;
|
|
|
|
shearcos = NSA_VALUE_SHEARCOS;
|
2011-04-22 18:53:56 +04:00
|
|
|
}
|
2009-01-20 19:53:06 +03:00
|
|
|
break;
|
|
|
|
}
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2009-07-24 03:05:34 +04:00
|
|
|
/* Scale to 16.16 fixed point */
|
2011-03-08 21:49:11 +03:00
|
|
|
ysize = fstyle->size * ((1 << 16) / FONT_SIZE_SCALE);
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
ofont = node->font;
|
|
|
|
|
2008-08-30 20:55:25 +04:00
|
|
|
if(ESetInfo(&ofont->olf_EEngine,
|
2011-04-22 20:06:46 +04:00
|
|
|
OT_DeviceDPI, ami_devicedpi,
|
2011-04-22 18:53:56 +04:00
|
|
|
OT_PointHeight, ysize,
|
2011-04-22 19:30:16 +04:00
|
|
|
OT_EmboldenX, emboldenx,
|
|
|
|
OT_EmboldenY, emboldeny,
|
|
|
|
OT_ShearSin, shearsin,
|
|
|
|
OT_ShearCos, shearcos,
|
2009-01-16 23:05:21 +03:00
|
|
|
TAG_END) == OTERR_Success)
|
|
|
|
return ofont;
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2009-01-16 23:05:21 +03:00
|
|
|
return NULL;
|
2008-08-30 20:55:25 +04:00
|
|
|
}
|
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
int32 ami_font_plot_glyph(struct OutlineFont *ofont, struct RastPort *rp,
|
|
|
|
uint16 char1, uint16 char2, uint32 x, uint32 y, uint32 emwidth)
|
|
|
|
{
|
|
|
|
struct GlyphMap *glyph;
|
|
|
|
UBYTE *glyphbm;
|
|
|
|
int32 char_advance = 0;
|
|
|
|
FIXED kern = 0;
|
|
|
|
|
|
|
|
if(ESetInfo(&ofont->olf_EEngine,
|
|
|
|
OT_GlyphCode, char1,
|
|
|
|
OT_GlyphCode2, char2,
|
|
|
|
TAG_END) == OTERR_Success)
|
|
|
|
{
|
|
|
|
if(EObtainInfo(&ofont->olf_EEngine,
|
|
|
|
OT_GlyphMap8Bit,&glyph,
|
|
|
|
TAG_END) == 0)
|
|
|
|
{
|
|
|
|
glyphbm = glyph->glm_BitMap;
|
|
|
|
if(!glyphbm) return 0;
|
|
|
|
|
|
|
|
if(rp)
|
|
|
|
{
|
|
|
|
BltBitMapTags(BLITA_SrcX, glyph->glm_BlackLeft,
|
|
|
|
BLITA_SrcY, glyph->glm_BlackTop,
|
|
|
|
BLITA_DestX, x - glyph->glm_X0 + glyph->glm_BlackLeft,
|
|
|
|
BLITA_DestY, y - glyph->glm_Y0 + glyph->glm_BlackTop,
|
|
|
|
BLITA_Width, glyph->glm_BlackWidth,
|
|
|
|
BLITA_Height, glyph->glm_BlackHeight,
|
|
|
|
BLITA_Source, glyphbm,
|
|
|
|
BLITA_SrcType, BLITT_ALPHATEMPLATE,
|
|
|
|
BLITA_Dest, rp,
|
|
|
|
BLITA_DestType, BLITT_RASTPORT,
|
|
|
|
BLITA_SrcBytesPerRow, glyph->glm_BMModulo,
|
|
|
|
TAG_DONE);
|
|
|
|
}
|
|
|
|
|
|
|
|
kern = 0;
|
|
|
|
|
|
|
|
if(char2) EObtainInfo(&ofont->olf_EEngine,
|
|
|
|
OT_TextKernPair, &kern,
|
|
|
|
TAG_END);
|
|
|
|
|
|
|
|
char_advance = (ULONG)(((glyph->glm_Width - kern) * emwidth) / 65536);
|
|
|
|
|
|
|
|
EReleaseInfo(&ofont->olf_EEngine,
|
|
|
|
OT_GlyphMap8Bit,glyph,
|
|
|
|
TAG_END);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return char_advance;
|
|
|
|
}
|
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
ULONG ami_unicode_text(struct RastPort *rp,const char *string,ULONG length,const plot_font_style_t *fstyle,ULONG dx, ULONG dy)
|
2008-08-30 20:55:25 +04:00
|
|
|
{
|
2009-01-24 15:24:15 +03:00
|
|
|
uint16 *utf16 = NULL, *outf16 = NULL;
|
2010-04-30 02:30:21 +04:00
|
|
|
uint16 utf16next = 0;
|
|
|
|
int utf16charlen;
|
2010-05-06 14:02:58 +04:00
|
|
|
struct OutlineFont *ofont, *ufont = NULL;
|
2008-08-30 20:55:25 +04:00
|
|
|
ULONG i,gx,gy;
|
|
|
|
UWORD posn;
|
2010-05-06 14:02:58 +04:00
|
|
|
uint32 x=0;
|
2009-07-03 03:07:04 +04:00
|
|
|
uint8 co = 0;
|
2010-05-06 14:02:58 +04:00
|
|
|
int32 tempx = 0;
|
2011-05-07 02:42:37 +04:00
|
|
|
ULONG emwidth = (ULONG)NSA_FONT_EMWIDTH(fstyle->size);
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2009-01-17 03:43:51 +03:00
|
|
|
if(!string || string[0]=='\0') return 0;
|
|
|
|
if(!length) return 0;
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2009-04-18 20:55:59 +04:00
|
|
|
if(utf8_to_enc(string,"UTF-16",length,(char **)&utf16) != UTF8_CONVERT_OK) return 0;
|
2009-01-24 15:24:15 +03:00
|
|
|
outf16 = utf16;
|
2010-05-06 14:02:58 +04:00
|
|
|
if(!(ofont = ami_open_outline_font(fstyle, FALSE))) return 0;
|
2008-08-30 20:55:25 +04:00
|
|
|
|
2009-07-21 14:59:53 +04:00
|
|
|
if(rp) SetRPAttrs(rp,RPTAG_APenColor,p96EncodeColor(RGBFB_A8B8G8R8,fstyle->foreground),TAG_DONE);
|
2009-01-17 03:43:51 +03:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
while(*utf16 != 0)
|
2008-08-30 20:55:25 +04:00
|
|
|
{
|
2010-04-30 02:30:21 +04:00
|
|
|
if (*utf16 < 0xD800 || 0xDFFF < *utf16)
|
|
|
|
utf16charlen = 1;
|
|
|
|
else
|
|
|
|
utf16charlen = 2;
|
|
|
|
|
2010-04-30 03:05:52 +04:00
|
|
|
utf16next = utf16[utf16charlen];
|
2010-04-30 02:30:21 +04:00
|
|
|
|
2010-05-06 14:02:58 +04:00
|
|
|
tempx = ami_font_plot_glyph(ofont, rp, *utf16, utf16next, dx + x, dy, emwidth);
|
|
|
|
|
|
|
|
if(tempx == 0)
|
2008-08-30 20:55:25 +04:00
|
|
|
{
|
2010-05-06 14:02:58 +04:00
|
|
|
if(ufont == NULL)
|
|
|
|
{
|
|
|
|
ufont = ami_open_outline_font(fstyle, TRUE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ufont)
|
|
|
|
{
|
|
|
|
tempx = ami_font_plot_glyph(ufont, rp, *utf16, utf16next,
|
|
|
|
dx + x, dy, emwidth);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
if(tempx == 0)
|
2008-08-30 20:55:25 +04:00
|
|
|
{
|
2010-05-06 14:02:58 +04:00
|
|
|
tempx = ami_font_plot_glyph(ofont, rp, 0xfffd, utf16next,
|
|
|
|
dx + x, dy, emwidth);
|
2008-08-30 20:55:25 +04:00
|
|
|
}
|
2010-05-06 14:02:58 +04:00
|
|
|
*/
|
2008-08-30 20:55:25 +04:00
|
|
|
}
|
2010-05-06 14:02:58 +04:00
|
|
|
|
|
|
|
x += tempx;
|
|
|
|
|
2010-04-30 02:30:21 +04:00
|
|
|
utf16 += utf16charlen;
|
2008-08-30 20:55:25 +04:00
|
|
|
}
|
|
|
|
|
2009-01-24 15:24:15 +03:00
|
|
|
free(outf16);
|
2009-01-16 23:05:21 +03:00
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ami_init_fonts(void)
|
|
|
|
{
|
2011-04-30 23:05:11 +04:00
|
|
|
ami_font_list = NewObjList();
|
2011-12-04 05:10:19 +04:00
|
|
|
NewList(&ami_diskfontlib_list);
|
2011-05-01 15:59:08 +04:00
|
|
|
|
|
|
|
/* run first cleanup in ten minutes */
|
2011-05-02 14:33:16 +04:00
|
|
|
schedule(60000, (schedule_callback_fn)ami_font_cleanup, ami_font_list);
|
2009-01-16 23:05:21 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ami_close_fonts(void)
|
|
|
|
{
|
2011-05-01 12:49:00 +04:00
|
|
|
LOG(("Cleaning up font cache"));
|
2011-04-30 23:05:11 +04:00
|
|
|
FreeObjList(ami_font_list);
|
|
|
|
ami_font_list = NULL;
|
|
|
|
}
|
2009-01-16 23:05:21 +03:00
|
|
|
|
2011-04-30 23:05:11 +04:00
|
|
|
void ami_font_close(struct ami_font_node *node)
|
|
|
|
{
|
|
|
|
/* Called from FreeObjList if node type is AMINS_FONT */
|
|
|
|
|
2011-12-04 05:10:19 +04:00
|
|
|
CloseOutlineFont(node->font, &ami_diskfontlib_list);
|
2008-08-30 20:55:25 +04:00
|
|
|
}
|
2011-04-22 20:06:46 +04:00
|
|
|
|
2011-05-01 15:59:08 +04:00
|
|
|
static void ami_font_cleanup(struct MinList *ami_font_list)
|
|
|
|
{
|
|
|
|
struct nsObject *node;
|
|
|
|
struct nsObject *nnode;
|
|
|
|
struct ami_font_node *fnode;
|
|
|
|
struct TimeVal curtime;
|
|
|
|
|
|
|
|
if(IsMinListEmpty(ami_font_list)) return;
|
|
|
|
|
|
|
|
node = (struct nsObject *)GetHead((struct List *)ami_font_list);
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
nnode=(struct nsObject *)GetSucc((struct Node *)node);
|
|
|
|
fnode = node->objstruct;
|
|
|
|
GetSysTime(&curtime);
|
|
|
|
SubTime(&curtime, &fnode->lastused);
|
|
|
|
if(curtime.Seconds > 300)
|
|
|
|
{
|
|
|
|
LOG(("Freeing %s not used for %d seconds",
|
|
|
|
node->dtz_Node.ln_Name, curtime.Seconds));
|
|
|
|
DelObject(node);
|
|
|
|
}
|
|
|
|
}while(node=nnode);
|
|
|
|
|
|
|
|
/* reschedule to run in five minutes */
|
2011-05-02 14:33:16 +04:00
|
|
|
schedule(30000, (schedule_callback_fn)ami_font_cleanup, ami_font_list);
|
2011-05-01 15:59:08 +04:00
|
|
|
}
|
|
|
|
|
2011-04-22 20:48:29 +04:00
|
|
|
void ami_font_setdevicedpi(int id)
|
2011-04-22 20:06:46 +04:00
|
|
|
{
|
2011-04-22 20:48:29 +04:00
|
|
|
DisplayInfoHandle dih;
|
|
|
|
struct DisplayInfo dinfo;
|
2011-04-25 21:43:22 +04:00
|
|
|
ULONG ydpi = option_amiga_ydpi;
|
|
|
|
ULONG xdpi = option_amiga_ydpi;
|
|
|
|
|
|
|
|
nscss_screen_dpi = INTTOFIX(option_amiga_ydpi);
|
2011-04-22 20:48:29 +04:00
|
|
|
|
2011-10-11 21:58:14 +04:00
|
|
|
if(id && (option_monitor_aspect_x != 0) && (option_monitor_aspect_y != 0))
|
2011-04-22 20:48:29 +04:00
|
|
|
{
|
|
|
|
if(dih = FindDisplayInfo(id))
|
|
|
|
{
|
|
|
|
if(GetDisplayInfoData(dih, &dinfo, sizeof(struct DisplayInfo),
|
|
|
|
DTAG_DISP, 0))
|
|
|
|
{
|
2011-07-17 18:48:19 +04:00
|
|
|
int xres = dinfo.Resolution.x;
|
|
|
|
int yres = dinfo.Resolution.y;
|
2011-04-22 20:48:29 +04:00
|
|
|
|
2011-10-11 21:58:14 +04:00
|
|
|
if((option_monitor_aspect_x != 4) || (option_monitor_aspect_y != 3))
|
2011-07-17 18:48:19 +04:00
|
|
|
{
|
|
|
|
/* AmigaOS sees 4:3 modes as square in the DisplayInfo database,
|
2011-10-11 21:58:14 +04:00
|
|
|
* so we correct other modes to "4:3 equiv" here. */
|
|
|
|
xres = (xres * option_monitor_aspect_x) / 4;
|
|
|
|
yres = (yres * option_monitor_aspect_y) / 3;
|
2011-07-17 18:48:19 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
xdpi = (yres * ydpi) / xres;
|
|
|
|
|
|
|
|
LOG(("XDPI = %ld, YDPI = %ld (DisplayInfo resolution %ld x %ld, corrected %ld x %ld)",
|
|
|
|
xdpi, ydpi, dinfo.Resolution.x, dinfo.Resolution.y, xres, yres));
|
2011-04-22 20:48:29 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-04-22 20:06:46 +04:00
|
|
|
|
2011-05-02 14:33:16 +04:00
|
|
|
ami_xdpi = xdpi;
|
2011-04-22 20:48:29 +04:00
|
|
|
ami_devicedpi = (xdpi << 16) | ydpi;
|
2011-04-22 20:06:46 +04:00
|
|
|
}
|
2011-09-08 03:01:53 +04:00
|
|
|
|
|
|
|
/* The below are simple font routines which should not be used for page rendering */
|
|
|
|
|
|
|
|
struct TextFont *ami_font_open_disk_font(struct TextAttr *tattr)
|
|
|
|
{
|
|
|
|
struct TextFont *tfont = OpenDiskFont(tattr);
|
|
|
|
return tfont;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ami_font_close_disk_font(struct TextFont *tfont)
|
|
|
|
{
|
|
|
|
CloseFont(tfont);
|
|
|
|
}
|