2015-08-17 02:16:17 +02:00
|
|
|
#ifndef INCLUDE_FONT_H
|
|
|
|
#define INCLUDE_FONT_H
|
|
|
|
|
|
|
|
#ifndef INCLUDE_MATH_H
|
|
|
|
#include "../lib/math.h"
|
|
|
|
#endif
|
|
|
|
|
2015-11-26 11:53:50 +01:00
|
|
|
#ifndef INCLUDE_FS_H
|
2015-11-06 15:49:37 +01:00
|
|
|
#include "../lib/obj/fs.h"
|
2015-08-17 02:16:17 +02:00
|
|
|
#endif
|
|
|
|
|
2015-12-17 20:15:49 +01:00
|
|
|
#define DEFAULT_FONT "/sys/fonts/Tahoma.kf"
|
|
|
|
|
2015-09-05 11:16:38 +02:00
|
|
|
:struct __OFFSET_FONT
|
|
|
|
{
|
|
|
|
signed x,y;
|
|
|
|
};
|
2015-08-20 17:04:18 +02:00
|
|
|
:struct __SIZE
|
|
|
|
{
|
2015-09-07 15:07:29 +02:00
|
|
|
dword width,height;
|
2015-09-05 11:16:38 +02:00
|
|
|
__OFFSET_FONT offset;
|
2015-08-21 13:03:08 +02:00
|
|
|
float offset_i,w_italic;
|
2015-08-20 17:04:18 +02:00
|
|
|
byte text;
|
2015-08-20 18:37:49 +02:00
|
|
|
byte TMP_WEIGHT;
|
2015-08-20 17:04:18 +02:00
|
|
|
};
|
2015-08-17 02:16:17 +02:00
|
|
|
:struct FONT
|
|
|
|
{
|
2015-08-20 17:04:18 +02:00
|
|
|
__SIZE size;
|
2015-12-17 20:15:49 +01:00
|
|
|
int left,top,width,height;
|
|
|
|
byte bold,italic,smooth;
|
|
|
|
dword bg_color;
|
2015-08-20 17:04:18 +02:00
|
|
|
dword color;
|
2015-08-17 18:21:51 +02:00
|
|
|
dword file_size;
|
2015-08-18 03:45:23 +02:00
|
|
|
dword buffer;
|
2015-08-18 11:13:34 +02:00
|
|
|
dword buffer_size;
|
2015-08-17 02:16:17 +02:00
|
|
|
word block;
|
|
|
|
dword data;
|
2015-08-17 18:21:51 +02:00
|
|
|
dword begin;
|
2015-08-17 02:16:17 +02:00
|
|
|
byte load(...);
|
2015-12-18 16:21:19 +01:00
|
|
|
byte changeSIZE();
|
|
|
|
byte symbol(signed x,y;byte s;dword c);
|
2015-08-18 12:43:23 +02:00
|
|
|
byte symbol_size(byte s);
|
2015-08-20 17:04:18 +02:00
|
|
|
dword getsize(dword text1);
|
2015-12-18 16:21:19 +01:00
|
|
|
void apply_smooth();
|
|
|
|
int write_center(dword x,y,w,h;dword txt);
|
|
|
|
int write(int x,y;dword text1);
|
|
|
|
void write_buf(int x,y,w,h, text1);
|
|
|
|
void show_buf();
|
2015-08-17 02:16:17 +02:00
|
|
|
};
|
|
|
|
FONT font = 0;
|
2015-12-17 20:15:49 +01:00
|
|
|
|
2015-08-20 17:04:18 +02:00
|
|
|
:byte FONT::changeSIZE()
|
2015-08-17 18:21:51 +02:00
|
|
|
{
|
|
|
|
dword TMP_DATA;
|
|
|
|
dword ofs;
|
2015-08-20 17:04:18 +02:00
|
|
|
IF(size.text<9) size.text = 8;
|
2015-09-01 15:55:35 +02:00
|
|
|
TMP_DATA = data = begin;
|
|
|
|
TMP_DATA +=size.text-8*4;
|
2015-08-20 18:37:49 +02:00
|
|
|
ofs = DSDWORD[TMP_DATA];
|
|
|
|
IF(ofs==-1)return false;
|
2015-09-01 15:55:35 +02:00
|
|
|
data += ofs + 156;
|
2015-08-20 18:37:49 +02:00
|
|
|
TMP_DATA = data;
|
|
|
|
file_size = DSDWORD[TMP_DATA];
|
2015-09-01 15:55:35 +02:00
|
|
|
TMP_DATA = data + file_size;
|
|
|
|
height = DSBYTE[TMP_DATA - 1];
|
|
|
|
width = DSBYTE[TMP_DATA - 2];
|
2015-08-20 18:37:49 +02:00
|
|
|
block = math.ceil(height*width/32);
|
|
|
|
return true;
|
2015-08-17 18:21:51 +02:00
|
|
|
}
|
2015-08-20 17:04:18 +02:00
|
|
|
:dword FONT::getsize(dword text1)
|
2015-08-17 02:16:17 +02:00
|
|
|
{
|
2015-08-20 17:04:18 +02:00
|
|
|
size.height = size.width = 0;
|
2015-09-05 11:16:38 +02:00
|
|
|
size.offset.x = size.offset.y = -1;
|
2015-08-20 17:04:18 +02:00
|
|
|
IF(size.text)IF(!changeSIZE())return 0;
|
2015-08-18 12:43:23 +02:00
|
|
|
WHILE(DSBYTE[text1])
|
|
|
|
{
|
2015-08-20 17:04:18 +02:00
|
|
|
symbol_size(DSBYTE[text1]);
|
2015-08-18 12:43:23 +02:00
|
|
|
text1++;
|
|
|
|
}
|
2015-09-05 11:16:38 +02:00
|
|
|
$neg size.offset.y
|
|
|
|
$neg size.offset.x
|
|
|
|
size.height += size.offset.y; size.height++;
|
|
|
|
size.width += size.offset.x; size.width++;
|
2015-08-21 13:03:08 +02:00
|
|
|
IF(italic)
|
|
|
|
{
|
2015-08-21 13:13:10 +02:00
|
|
|
size.w_italic = size.height/3;
|
|
|
|
size.offset_i = size.w_italic/size.height;
|
|
|
|
size.width += size.w_italic;
|
|
|
|
size.w_italic = -size.w_italic;
|
2015-08-21 13:03:08 +02:00
|
|
|
}
|
2015-08-20 17:04:18 +02:00
|
|
|
return size.width;
|
2015-08-18 12:43:23 +02:00
|
|
|
}
|
|
|
|
:byte FONT::symbol_size(byte s)
|
|
|
|
{
|
2015-12-17 20:15:49 +01:00
|
|
|
dword xi,yi;
|
|
|
|
dword tmp,_;
|
2015-12-18 16:21:19 +01:00
|
|
|
dword iii = 0;
|
2015-12-17 20:15:49 +01:00
|
|
|
byte rw=0;
|
2015-08-20 17:04:18 +02:00
|
|
|
byte X;
|
2015-08-20 18:37:49 +02:00
|
|
|
size.TMP_WEIGHT = math.ceil(size.text/17);
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(s==32)
|
2015-08-20 17:04:18 +02:00
|
|
|
{
|
|
|
|
size.width += width/4;
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(bold) size.width+=size.TMP_WEIGHT;
|
2015-08-20 17:04:18 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
IF(s==9)
|
|
|
|
{
|
|
|
|
size.width += width;
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(bold) size.width+=size.TMP_WEIGHT;
|
2015-08-20 17:04:18 +02:00
|
|
|
return;
|
|
|
|
}
|
2015-12-17 20:15:49 +01:00
|
|
|
s = AnsiToCp866(s);
|
|
|
|
tmp = 4*block*s + data;
|
|
|
|
for(yi=0; yi<height; yi++)
|
|
|
|
{
|
|
|
|
for(xi=0; xi<width; xi++)
|
|
|
|
{
|
|
|
|
IF(iii%32) _ >>= 1;
|
2015-09-02 15:15:32 +02:00
|
|
|
ELSE
|
|
|
|
{
|
2015-12-17 20:15:49 +01:00
|
|
|
tmp += 4;
|
|
|
|
_ = DSDWORD[tmp];
|
|
|
|
}
|
|
|
|
IF(_&1)
|
2015-09-02 15:15:32 +02:00
|
|
|
{
|
|
|
|
IF(xi>rw)rw=xi;
|
|
|
|
IF(size.height<yi)size.height = yi;
|
2015-09-05 11:16:38 +02:00
|
|
|
IF(size.offset.y<0)size.offset.y = yi;
|
|
|
|
ELSE IF(yi<size.offset.y)size.offset.y = yi;
|
2015-09-02 15:15:32 +02:00
|
|
|
IF(!X) X = xi;
|
|
|
|
ELSE IF(X>xi)X = xi;
|
|
|
|
}
|
2015-12-17 20:15:49 +01:00
|
|
|
iii++;
|
|
|
|
}
|
|
|
|
}
|
2015-08-20 17:04:18 +02:00
|
|
|
size.width += rw;
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(bold) size.width+=size.TMP_WEIGHT;
|
2015-08-20 17:04:18 +02:00
|
|
|
IF(s=='_') size.width--;
|
2015-09-05 11:16:38 +02:00
|
|
|
IF(size.offset.x<0)size.offset.x = X;
|
2015-08-18 12:43:23 +02:00
|
|
|
}
|
2015-12-17 20:15:49 +01:00
|
|
|
:byte FONT::symbol(signed x,y;byte s)
|
|
|
|
{
|
|
|
|
dword xi,yi;
|
2015-12-18 16:21:19 +01:00
|
|
|
dword iii = 0;
|
|
|
|
dword offs;
|
2015-12-17 20:15:49 +01:00
|
|
|
float ital = -size.w_italic;
|
|
|
|
dword ___x;
|
|
|
|
byte rw=0;
|
|
|
|
IF(s==32)return width/4;
|
|
|
|
IF(s==9)return width;
|
|
|
|
s = AnsiToCp866(s);
|
|
|
|
EBX = block*s << 2 + data;
|
|
|
|
for(yi=0; yi<height; yi++)
|
|
|
|
{
|
2015-12-18 16:21:19 +01:00
|
|
|
EDI = size.offset.y + yi + y * size.width * 3 + buffer;
|
2015-12-17 20:15:49 +01:00
|
|
|
for(xi=0; xi<width; xi++)
|
|
|
|
{
|
|
|
|
IF(iii%32) $shr ecx,1
|
|
|
|
ELSE
|
|
|
|
{
|
|
|
|
EBX += 4;
|
|
|
|
ECX = DSDWORD[EBX];
|
|
|
|
}
|
|
|
|
IF(ECX&true)
|
|
|
|
{
|
|
|
|
IF(xi>rw)rw=xi;
|
|
|
|
___x = x+xi;
|
|
|
|
IF(italic)___x+=math.ceil(ital);
|
2015-12-18 16:21:19 +01:00
|
|
|
offs = ___x*3 + EDI;
|
|
|
|
DSDWORD[offs] = DSDWORD[offs] & 0xFF000000 | color;
|
|
|
|
IF(bold) DSDWORD[offs+3] = DSDWORD[offs+3] & 0xFF000000 | color;
|
2015-12-17 20:15:49 +01:00
|
|
|
}
|
|
|
|
iii++;
|
|
|
|
}
|
|
|
|
if (italic) ital-=size.offset_i;
|
|
|
|
}
|
|
|
|
return rw;
|
|
|
|
}
|
|
|
|
|
|
|
|
byte AnsiToCp866(byte s) {
|
|
|
|
IF(s>=128)&&(s<=175)s+=64;
|
|
|
|
ELSE IF(s>=224)&&(s<=239)s+=16;
|
|
|
|
ELSE IF(s==241)s=184; //yo
|
|
|
|
ELSE IF(s==240)s=168; //YO
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline fastcall dword b24(EBX) { return DSDWORD[EBX] << 8; }
|
2015-12-18 16:21:19 +01:00
|
|
|
:void FONT::apply_smooth()
|
2015-12-17 20:15:49 +01:00
|
|
|
{
|
|
|
|
dword i,line_w,to;
|
2015-12-18 16:21:19 +01:00
|
|
|
line_w = font.size.width * 3;
|
|
|
|
to = font.size.height - 1 * line_w + font.buffer - 3;
|
|
|
|
for(i=font.buffer; i < to; i+=3)
|
|
|
|
{
|
|
|
|
IF(i-font.buffer%line_w +3 == line_w) continue;
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(b24(i)==0x000000) && (b24(i+3)!=0x000000) && (b24(i+line_w)!=0x000000) && (b24(i+3+line_w)==0x000000)
|
|
|
|
{
|
2015-12-18 16:21:19 +01:00
|
|
|
ShadowPixel(i+3, 2);
|
|
|
|
ShadowPixel(i+line_w, 2);
|
2015-12-17 20:15:49 +01:00
|
|
|
}
|
|
|
|
ELSE IF(b24(i)!=0x000000) && (b24(i+3)==0x000000) && (b24(i+line_w)==0x000000) && (b24(i+3+line_w)!=0x000000)
|
|
|
|
{
|
2015-12-18 16:21:19 +01:00
|
|
|
ShadowPixel(i, 2);
|
|
|
|
ShadowPixel(i+3+line_w, 2);
|
2015-12-17 20:15:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
:byte FONT::load(dword path)
|
|
|
|
{
|
|
|
|
lib_init_fs();
|
|
|
|
buffer_size = 0;
|
|
|
|
smooth = true;
|
|
|
|
IF(data)free(data);
|
|
|
|
IF(!fs.read(path)) { debug("Error while loading font: "); debugln(path); return false; }
|
|
|
|
begin = data = EAX;
|
|
|
|
EBX = begin + ECX;
|
|
|
|
height = DSBYTE[EBX-1];
|
|
|
|
width = DSBYTE[EBX-2];
|
|
|
|
block = math.ceil(height*width/32);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-18 16:21:19 +01:00
|
|
|
:int FONT::write_center(dword x,y,w,h;dword txt)
|
|
|
|
{
|
|
|
|
getsize(txt);
|
|
|
|
return write(w-size.width/2+x,y,txt);
|
|
|
|
}
|
|
|
|
|
|
|
|
:int FONT::write(int x,y; dword text1)
|
2015-08-18 12:43:23 +02:00
|
|
|
{
|
|
|
|
signed len=0;
|
2015-08-20 18:37:49 +02:00
|
|
|
IF(!text1)return false;
|
|
|
|
IF(size.text)IF(!changeSIZE())return false;
|
2015-12-17 20:15:49 +01:00
|
|
|
left = x;
|
2015-08-20 17:04:18 +02:00
|
|
|
getsize(text1);
|
2015-09-05 11:16:38 +02:00
|
|
|
y -= size.offset.y;
|
2015-09-07 13:33:25 +02:00
|
|
|
top = y;
|
2015-08-20 18:37:49 +02:00
|
|
|
EDX = size.width*size.height*3;
|
2015-08-19 13:14:15 +02:00
|
|
|
IF(!buffer_size)
|
|
|
|
{
|
2015-08-20 18:37:49 +02:00
|
|
|
buffer_size = EDX;
|
2015-08-19 13:14:15 +02:00
|
|
|
buffer = malloc(buffer_size);
|
|
|
|
}
|
2015-08-20 18:37:49 +02:00
|
|
|
ELSE IF(buffer_size<EDX)
|
2015-08-19 13:14:15 +02:00
|
|
|
{
|
2015-08-20 18:37:49 +02:00
|
|
|
buffer_size = EDX;
|
2015-08-19 13:14:15 +02:00
|
|
|
buffer = realloc(buffer,buffer_size);
|
|
|
|
}
|
2015-12-18 16:21:19 +01:00
|
|
|
// Fill background color {
|
|
|
|
EBX = bg_color;
|
|
|
|
EAX = buffer_size+buffer;
|
|
|
|
for (EDI=buffer; EDI<EAX; EDI+=3) ESDWORD[EDI] = EBX;
|
|
|
|
// }
|
2015-09-05 11:16:38 +02:00
|
|
|
len = size.offset.x;
|
2015-08-17 02:16:17 +02:00
|
|
|
WHILE(DSBYTE[text1])
|
|
|
|
{
|
2015-08-20 17:04:18 +02:00
|
|
|
IF(DSBYTE[text1]=='_') len--;
|
2015-09-02 15:15:32 +02:00
|
|
|
len+=symbol(len,0,DSBYTE[text1]);
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(bold)len+=math.ceil(size.text/17);
|
2015-08-17 02:16:17 +02:00
|
|
|
text1++;
|
|
|
|
}
|
2015-12-18 16:21:19 +01:00
|
|
|
IF (smooth) apply_smooth();
|
|
|
|
show_buf(left,top);
|
2015-08-17 02:16:17 +02:00
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2015-12-18 16:21:19 +01:00
|
|
|
:void FONT::write_buf(int x,y,w,h; dword text1)
|
2015-09-01 15:11:07 +02:00
|
|
|
{
|
2015-12-18 16:21:19 +01:00
|
|
|
dword new_buffer_size;
|
2015-09-01 15:11:07 +02:00
|
|
|
IF(!text1)return;
|
|
|
|
IF(size.text)IF(!changeSIZE())return;
|
|
|
|
getsize(text1);
|
2015-09-05 11:16:38 +02:00
|
|
|
y -= size.offset.y;
|
2015-12-17 20:15:49 +01:00
|
|
|
|
2015-09-01 15:11:07 +02:00
|
|
|
size.width = w;
|
2015-09-01 15:55:35 +02:00
|
|
|
size.height = h;
|
2015-09-02 15:15:32 +02:00
|
|
|
|
2015-09-07 15:07:29 +02:00
|
|
|
new_buffer_size = w*h*3;
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(buffer_size != w*h*3)
|
2015-09-01 15:11:07 +02:00
|
|
|
{
|
2015-09-07 15:07:29 +02:00
|
|
|
buffer_size = new_buffer_size;
|
2015-09-02 15:15:32 +02:00
|
|
|
free(buffer);
|
2015-09-07 15:07:29 +02:00
|
|
|
buffer = malloc(buffer_size);
|
2015-12-18 16:21:19 +01:00
|
|
|
// Fill background color
|
2015-09-07 15:07:29 +02:00
|
|
|
EBX = bg_color;
|
|
|
|
EAX = buffer_size+buffer;
|
2015-12-17 20:15:49 +01:00
|
|
|
for (EDI=buffer; EDI<EAX; EDI+=3) ESDWORD[EDI] = EBX;
|
2015-09-01 15:11:07 +02:00
|
|
|
}
|
|
|
|
WHILE(DSBYTE[text1])
|
|
|
|
{
|
2015-09-02 15:15:32 +02:00
|
|
|
x+=symbol(x,y,DSBYTE[text1]);
|
2015-12-17 20:15:49 +01:00
|
|
|
IF(bold)x+=math.ceil(size.text/17);
|
2015-09-01 15:11:07 +02:00
|
|
|
text1++;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-12-18 16:21:19 +01:00
|
|
|
:void FONT::show_buf(dword left1, top1){
|
|
|
|
_PutImage(left1,top1,size.width,size.height,buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-08-17 02:16:17 +02:00
|
|
|
#endif
|