mirror of
https://github.com/encounter/mkdd.git
synced 2026-03-30 11:30:02 -07:00
256 lines
8.2 KiB
C++
256 lines
8.2 KiB
C++
#include <stdio.h>
|
|
|
|
#include <dolphin/os.h>
|
|
#include "JSystem/JUtility/JUTDirectPrint.h"
|
|
|
|
JUTDirectPrint *JUTDirectPrint::sDirectPrint;
|
|
|
|
JUTDirectPrint::JUTDirectPrint()
|
|
{
|
|
changeFrameBuffer(nullptr, 0, 0);
|
|
setCharColor(0xff, 0xff, 0xff);
|
|
}
|
|
|
|
JUTDirectPrint *JUTDirectPrint::start()
|
|
{
|
|
if (!sDirectPrint) {
|
|
sDirectPrint = new JUTDirectPrint();
|
|
}
|
|
|
|
return sDirectPrint;
|
|
}
|
|
|
|
void JUTDirectPrint::erase(int x, int y, int width, int height)
|
|
{
|
|
if (!mFramebuffer) {
|
|
return;
|
|
}
|
|
|
|
if (400 < mFbWidth)
|
|
{
|
|
x = x << 1;
|
|
width = width << 1;
|
|
}
|
|
|
|
if (300 < mFbHeight)
|
|
{
|
|
y = y << 1;
|
|
height = height << 1;
|
|
}
|
|
|
|
u16 *pixel = mFrameMemory + mStride * y + x;
|
|
for (int i = 0; i < height; i++)
|
|
{
|
|
for (int j = 0; j < width; j++)
|
|
{
|
|
*pixel = 0x1080;
|
|
pixel = pixel + 1;
|
|
}
|
|
|
|
pixel += mStride - width;
|
|
}
|
|
}
|
|
|
|
u8 JUTDirectPrint::sAsciiTable[128] = {
|
|
0x7A, 0x7A, 0x7A, 0x7A, 0x7A, 0x7A, 0x7A, 0x7A, 0x7A, 0xFD, 0xFE, 0x7A, 0x7A, 0x7A, 0x7A, 0x7A,
|
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
|
0xFF, 0x29, 0x64, 0x65, 0x66, 0x2B, 0x67, 0x68, 0x25, 0x26, 0x69, 0x2A, 0x6A, 0x27, 0x2C, 0x6B,
|
|
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x24, 0x6C, 0x6D, 0x6E, 0x6F, 0x28,
|
|
0x70, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18,
|
|
0x19, 0x1A, 0x1B, 0x1C, 0x1D, 0x1E, 0x1F, 0x20, 0x21, 0x22, 0x23, 0x71, 0x72, 0x73, 0x74, 0x75,
|
|
0xFF, 0x7D, 0x7E, 0x7F, 0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8A, 0x8B,
|
|
0x8C, 0x8D, 0x8E, 0x8F, 0x90, 0x91, 0x92, 0x93, 0x94, 0x95, 0x96, 0x76, 0x77, 0x78, 0x79, 0x7A,
|
|
};
|
|
|
|
u32 JUTDirectPrint::sFontData[64] = {
|
|
0x70871C30, 0x8988A250, 0x88808290, 0x88830C90, 0x888402F8, 0x88882210, 0x71CF9C10, 0xF9CF9C70,
|
|
0x8208A288, 0xF200A288, 0x0BC11C78, 0x0A222208, 0x8A222208, 0x71C21C70, 0x23C738F8, 0x5228A480,
|
|
0x8A282280, 0x8BC822F0, 0xFA282280, 0x8A28A480, 0x8BC738F8, 0xF9C89C08, 0x82288808, 0x82088808,
|
|
0xF2EF8808, 0x82288888, 0x82288888, 0x81C89C70, 0x8A08A270, 0x920DA288, 0xA20AB288, 0xC20AAA88,
|
|
0xA208A688, 0x9208A288, 0x8BE8A270, 0xF1CF1CF8, 0x8A28A220, 0x8A28A020, 0xF22F1C20, 0x82AA0220,
|
|
0x82492220, 0x81A89C20, 0x8A28A288, 0x8A28A288, 0x8A289488, 0x8A2A8850, 0x894A9420, 0x894AA220,
|
|
0x70852220, 0xF8011000, 0x08020800, 0x10840400, 0x20040470, 0x40840400, 0x80020800, 0xF8011000,
|
|
0x70800000, 0x88822200, 0x08820400, 0x108F8800, 0x20821000, 0x00022200, 0x20800020, 0x00000000,
|
|
};
|
|
|
|
u32 JUTDirectPrint::sFontData2[77] = {
|
|
0x51421820, 0x53E7A420, 0x014A2C40, 0x01471000, 0x0142AA00, 0x03EAA400, 0x01471A78, 0x00000000,
|
|
0x50008010, 0x20010820, 0xF8020040, 0x20420820, 0x50441010, 0x00880000, 0x00070E00, 0x01088840,
|
|
0x78898820, 0x004A8810, 0x788A8810, 0x01098808, 0x00040E04, 0x70800620, 0x11400820, 0x12200820,
|
|
0x10001020, 0x10000820, 0x100F8820, 0x70000620, 0x60070000, 0x110F82A0, 0x12AA8AE0, 0x084F92A0,
|
|
0x100FBE1C, 0x10089008, 0x60070808, 0x00000000, 0x02000200, 0x7A078270, 0x8BC81E88, 0x8A2822F8,
|
|
0x9A282280, 0x6BC79E78, 0x30000000, 0x48080810, 0x41E80000, 0x422F1830, 0xFBE88810, 0x40288890,
|
|
0x43C89C60, 0x81000000, 0x81000000, 0x990F3C70, 0xA10AA288, 0xE10AA288, 0xA10AA288, 0x98CAA270,
|
|
0x00000000, 0x00000020, 0xF1EF1E20, 0x8A28A0F8, 0x8A281C20, 0xF1E80220, 0x80283C38, 0x00000000,
|
|
0x00000000, 0x8A28B688, 0x8A2A8888, 0x8A2A8878, 0x894A8808, 0x788536F0, 0x00000000, 0x00000000,
|
|
0xF8000000, 0x10000000, 0x20000000, 0x40000000, 0xF8000000,
|
|
};
|
|
|
|
void JUTDirectPrint::drawChar(int position_x, int position_y, int ch)
|
|
{
|
|
static u32 twiceBit[4] = {
|
|
0x00000000,
|
|
0x00000003,
|
|
0x0000000C,
|
|
0x0000000F
|
|
};
|
|
|
|
int codepoint = (100 <= ch) ? ch - 100 : ch;
|
|
int col_index = (codepoint % 5) * 6;
|
|
int row_index = (codepoint / 5) * 7;
|
|
|
|
const u32 *font_data = (100 > ch) ? sFontData + row_index : sFontData2 + row_index;
|
|
|
|
int scale_x = (mFbWidth < 400) ? 1 : 2;
|
|
int scale_y = (mFbHeight < 300) ? 1 : 2;
|
|
|
|
u16 *pixel = mFrameMemory + mStride * position_y * scale_y + position_x * scale_x;
|
|
for (int y = 0; y < 7; y++)
|
|
{
|
|
u32 data = *font_data << col_index;
|
|
font_data += 1;
|
|
|
|
if (scale_x == 1)
|
|
{
|
|
data = (data & 0xfc000000) >> 1;
|
|
}
|
|
else
|
|
{
|
|
u32 a = twiceBit[(data >> 26) & 3];
|
|
u32 b = twiceBit[(data >> 28) & 3] << 4;
|
|
u32 c = twiceBit[(data >> 30) & 3] << 8;
|
|
data = (a | b | c) << 19;
|
|
}
|
|
|
|
for (int x = 0; x < scale_x * 6; x += 2)
|
|
{
|
|
u16 value;
|
|
|
|
value = (((data & 0x40000000) ? mCharColorY : 0) |
|
|
((data & 0x80000000) ? mCharColorCb4 : 32) +
|
|
((data & 0x40000000) ? mCharColorCb2 : 64) +
|
|
((data & 0x20000000) ? mCharColorCb4 : 32));
|
|
pixel[0] = value;
|
|
if (scale_y > 1)
|
|
pixel[mStride] = value;
|
|
|
|
value = (((data & 0x20000000) ? mCharColorY : 0) |
|
|
((data & 0x40000000) ? mCharColorCr4 : 32) +
|
|
((data & 0x20000000) ? mCharColorCr2 : 64) +
|
|
((data & 0x10000000) ? mCharColorCr4 : 32));
|
|
pixel[1] = value;
|
|
if (scale_y > 1)
|
|
pixel[1 + mStride] = value;
|
|
|
|
pixel += 2;
|
|
data <<= 2;
|
|
|
|
}
|
|
|
|
pixel += mStride * scale_y - 6 * scale_x;
|
|
}
|
|
}
|
|
|
|
void JUTDirectPrint::changeFrameBuffer(void *fb, u16 width, u16 height)
|
|
{
|
|
mFramebuffer = fb;
|
|
mFrameMemory = (u16 *)fb;
|
|
mFbWidth = width;
|
|
mFbHeight = height;
|
|
mStride = ALIGN_NEXT(width, 16);
|
|
mFbSize = (u32)mStride * (u32)mFbHeight * 2;
|
|
}
|
|
|
|
void JUTDirectPrint::printSub(u16 position_x, u16 position_y, const char *format, va_list args,
|
|
bool clear)
|
|
{
|
|
char buffer[256];
|
|
if (!mFrameMemory)
|
|
{
|
|
return;
|
|
}
|
|
|
|
int buffer_length = vsnprintf(buffer, ARRAY_SIZE(buffer), format, args);
|
|
u16 x = position_x;
|
|
if (buffer_length > 0)
|
|
{
|
|
if (clear)
|
|
{
|
|
erase(position_x - 6, position_y - 3, (buffer_length + 2) * 6, 0xd);
|
|
}
|
|
|
|
char *ptr = buffer;
|
|
for (; 0 < buffer_length; buffer_length--, ptr++)
|
|
{
|
|
int codepoint = sAsciiTable[*ptr & 0x7f];
|
|
if (codepoint == 0xfe)
|
|
{
|
|
position_x = x;
|
|
position_y += 7;
|
|
}
|
|
else if (codepoint == 0xfd)
|
|
{
|
|
s32 current_position = (int)position_x;
|
|
s32 tab = (current_position - x + 0x2f) % 0x30;
|
|
position_x = current_position + 0x30 - tab;
|
|
}
|
|
else
|
|
{
|
|
if (codepoint != 0xff)
|
|
{
|
|
drawChar(position_x, position_y, codepoint);
|
|
}
|
|
position_x += 6;
|
|
}
|
|
}
|
|
}
|
|
|
|
DCStoreRange(mFrameMemory, mFbSize);
|
|
}
|
|
|
|
void JUTDirectPrint::print(u16 position_x, u16 position_y, char const *format, ...)
|
|
{
|
|
if (mFrameMemory)
|
|
{
|
|
va_list args;
|
|
va_start(args, format);
|
|
printSub(position_x, position_y, format, args, true);
|
|
va_end(args);
|
|
}
|
|
}
|
|
|
|
void JUTDirectPrint::drawString(u16 position_x, u16 position_y, char *text)
|
|
{
|
|
drawString_f(position_x, position_y, "%s", text);
|
|
}
|
|
|
|
void JUTDirectPrint::drawString_f(u16 position_x, u16 position_y, char const *format, ...)
|
|
{
|
|
if (mFrameMemory)
|
|
{
|
|
va_list args;
|
|
va_start(args, format);
|
|
printSub(position_x, position_y, format, args, false);
|
|
va_end(args);
|
|
}
|
|
}
|
|
|
|
void JUTDirectPrint::setCharColor(JUtility::TColor color) {
|
|
setCharColor(color.r, color.g, color.b);
|
|
}
|
|
|
|
void JUTDirectPrint::setCharColor(u8 r, u8 g, u8 b)
|
|
{
|
|
mCharColor = JUtility::TColor(r, g, b, 0xFF);
|
|
u16 Y = 0.257 * (int)r + 0.504 * (int)g + 0.098 * (int)b + 16;
|
|
u16 Cb = -0.148 * (int)r - 0.291 * (int)g + 0.439 * (int)b + 128;
|
|
u16 Cr = 0.439 * (int)r - 0.368 * (int)g - 0.071 * (int)b + 128;
|
|
|
|
mCharColorY = Y << 8;
|
|
mCharColorCb = Cb;
|
|
mCharColorCb2 = Cb / 2;
|
|
mCharColorCb4 = Cb / 4;
|
|
mCharColorCr = Cr;
|
|
mCharColorCr2 = Cr / 2;
|
|
mCharColorCr4 = Cr / 4;
|
|
} |