1
0
mirror of https://github.com/VDR4Arch/vdr.git synced 2023-10-10 13:36:52 +02:00
vdr/font.c

236 lines
4.9 KiB
C
Raw Permalink Normal View History

2000-10-03 10:34:48 +02:00
/*
* font.c: Font handling for the DVB On Screen Display
*
* See the main source file 'vdr.c' for copyright information and
* how to reach the author.
*
2004-05-16 12:05:40 +02:00
* $Id: font.c 1.7 2004/05/16 10:50:59 kls Exp $
2000-10-03 10:34:48 +02:00
*/
2004-05-16 10:35:36 +02:00
#include "config.h"
#include <ctype.h>
2000-10-03 10:34:48 +02:00
#include "font.h"
#include "tools.h"
2000-11-18 15:46:00 +01:00
#include "fontfix.c"
2000-10-03 10:34:48 +02:00
#include "fontosd.c"
#include "fontsml.c"
2000-10-03 10:34:48 +02:00
2004-05-16 12:05:40 +02:00
#include "fontfix-iso8859-2.c"
#include "fontosd-iso8859-2.c"
#include "fontsml-iso8859-2.c"
2004-01-16 13:34:43 +01:00
#include "fontfix-iso8859-5.c"
#include "fontosd-iso8859-5.c"
#include "fontsml-iso8859-5.c"
#include "fontfix-iso8859-7.c"
#include "fontosd-iso8859-7.c"
#include "fontsml-iso8859-7.c"
2004-05-16 10:35:36 +02:00
// --- cFont -----------------------------------------------------------------
static void *FontData[eDvbCodeSize][eDvbFontSize] = {
{ FontOsd_iso8859_1, FontFix_iso8859_1, FontSml_iso8859_1 },
2004-05-16 12:05:40 +02:00
{ FontOsd_iso8859_2, FontFix_iso8859_2, FontSml_iso8859_2 },
2004-01-16 13:34:43 +01:00
{ FontOsd_iso8859_5, FontFix_iso8859_5, FontSml_iso8859_5 },
{ FontOsd_iso8859_7, FontFix_iso8859_7, FontSml_iso8859_7 },
};
static const char *FontCode[eDvbCodeSize] = {
"iso8859-1",
2004-05-16 12:05:40 +02:00
"iso8859-2",
2004-01-16 13:34:43 +01:00
"iso8859-5",
"iso8859-7",
};
2000-11-18 15:46:00 +01:00
eDvbCode cFont::code = code_iso8859_1;
cFont *cFont::fonts[eDvbFontSize] = { NULL };
2000-11-18 15:46:00 +01:00
cFont::cFont(void *Data)
{
SetData(Data);
2000-10-03 10:34:48 +02:00
}
void cFont::SetData(void *Data)
{
2004-05-16 10:35:36 +02:00
height = ((tCharData *)Data)->height;
for (int i = 0; i < NUMCHARS; i++)
2004-05-16 10:35:36 +02:00
data[i] = (tCharData *)&((tPixelData *)Data)[(i < 32 ? 0 : i - 32) * (height + 2)];
}
int cFont::Width(const char *s) const
2000-10-03 10:34:48 +02:00
{
int w = 0;
while (s && *s)
w += Width(*s++);
return w;
}
int cFont::Height(const char *s) const
2000-10-03 10:34:48 +02:00
{
int h = 0;
if (s && *s)
2004-05-16 10:35:36 +02:00
h = height; // all characters have the same height!
2000-10-03 10:34:48 +02:00
return h;
}
bool cFont::SetCode(const char *Code)
{
for (int i = 0; i < eDvbCodeSize; i++) {
if (strcmp(Code, FontCode[i]) == 0) {
SetCode(eDvbCode(i));
return true;
}
}
return false;
}
void cFont::SetCode(eDvbCode Code)
{
if (code != Code) {
code = Code;
for (int i = 0; i < eDvbFontSize; i++) {
if (fonts[i])
fonts[i]->SetData(FontData[code][i]);
}
}
}
void cFont::SetFont(eDvbFont Font, void *Data)
{
delete fonts[Font];
fonts[Font] = new cFont(Data ? Data : FontData[code][Font]);
}
const cFont *cFont::GetFont(eDvbFont Font)
{
2004-05-16 10:35:36 +02:00
if (Setup.UseSmallFont == 0 && Font == fontSml)
Font = fontOsd;
else if (Setup.UseSmallFont == 2 && Font == fontOsd)
Font = fontSml;
if (!fonts[Font])
SetFont(Font);
return fonts[Font];
}
2004-05-16 10:35:36 +02:00
// --- cTextWrapper ----------------------------------------------------------
cTextWrapper::cTextWrapper(void)
{
text = eol = NULL;
lines = 0;
lastLine = -1;
}
cTextWrapper::cTextWrapper(const char *Text, const cFont *Font, int Width)
{
text = NULL;
Set(Text, Font, Width);
}
cTextWrapper::~cTextWrapper()
{
free(text);
}
void cTextWrapper::Set(const char *Text, const cFont *Font, int Width)
{
free(text);
text = Text ? strdup(Text) : NULL;
eol = NULL;
lines = 0;
lastLine = -1;
if (!text)
return;
lines = 1;
if (Width <= 0)
return;
char *Blank = NULL;
char *Delim = NULL;
int w = 0;
stripspace(text); // strips trailing newlines
for (char *p = text; *p; ) {
if (*p == '\n') {
lines++;
w = 0;
Blank = Delim = NULL;
p++;
continue;
}
else if (isspace(*p))
Blank = p;
int cw = Font->Width(*p);
if (w + cw > Width) {
if (Blank) {
*Blank = '\n';
p = Blank;
continue;
}
else {
// Here's the ugly part, where we don't have any whitespace to
// punch in a newline, so we need to make room for it:
if (Delim)
p = Delim + 1; // let's fall back to the most recent delimiter
char *s = MALLOC(char, strlen(text) + 2); // The additional '\n' plus the terminating '\0'
int l = p - text;
strncpy(s, text, l);
s[l] = '\n';
strcpy(s + l + 1, p);
free(text);
text = s;
p = text + l;
continue;
}
}
else
w += cw;
if (strchr("-.,:;!?_", *p)) {
Delim = p;
Blank = NULL;
}
p++;
}
}
const char *cTextWrapper::Text(void)
{
if (eol) {
*eol = '\n';
eol = NULL;
}
return text;
}
const char *cTextWrapper::GetLine(int Line)
{
char *s = NULL;
if (Line < lines) {
if (eol) {
*eol = '\n';
if (Line == lastLine + 1)
s = eol + 1;
eol = NULL;
}
if (!s) {
s = text;
for (int i = 0; i < Line; i++) {
s = strchr(s, '\n');
if (s)
s++;
else
break;
}
}
if (s) {
if ((eol = strchr(s, '\n')) != NULL)
*eol = 0;
}
lastLine = Line;
}
return s;
}