project refactoring - win32 platform support

This commit is contained in:
Vadim Lopatin 2014-03-05 09:29:03 +04:00
parent 81833a6db6
commit 3d2fc6df46
6 changed files with 552 additions and 567 deletions

BIN
3rdparty/libpng/lib/libpng15.dll vendored Normal file

Binary file not shown.

BIN
3rdparty/libpng/lib/libpng15.lib vendored Normal file

Binary file not shown.

View File

@ -191,9 +191,6 @@
<Folder name="dlangui">
<Folder name="3rdparty">
<Folder name="libpng">
<Folder name="libs">
<File path="3rdparty\libpng\lib\libpng15.lib" />
</Folder>
<File path="3rdparty\libpng\source\libpng\png.d" />
<File path="3rdparty\libpng\source\libpng\pngconf.d" />
<File path="3rdparty\libpng\source\libpng\pnglibconf.d" />
@ -266,6 +263,8 @@
<File path="src\dlangui\platforms\common\platform.d" />
</Folder>
<Folder name="windows">
<File path="src\dlangui\platforms\windows\win32drawbuf.d" />
<File path="src\dlangui\platforms\windows\win32fonts.d" />
<File path="src\dlangui\platforms\windows\winapp.d" />
</Folder>
</Folder>

View File

@ -0,0 +1,69 @@
module dlangui.platforms.windows.win32drawbuf;
version (Windows) {
import win32.windows;
import dlangui.graphics.drawbuf;
class Win32ColorDrawBuf : ColorDrawBufBase {
uint * _pixels;
HDC _drawdc;
HBITMAP _drawbmp;
@property HDC dc() { return _drawdc; }
this(int width, int height) {
resize(width, height);
}
override uint * scanLine(int y) {
if (y >= 0 && y < _dy)
return _pixels + _dx * (_dy - 1 - y);
return null;
}
override void clear() {
if (_drawbmp !is null) {
DeleteObject( _drawbmp );
DeleteObject( _drawdc );
_pixels = null;
_dx = 0;
_dy = 0;
}
}
override void resize(int width, int height) {
if (width< 0)
width = 0;
if (height < 0)
height = 0;
if (_dx == width && _dy == height)
return;
clear();
_dx = width;
_dy = height;
if (_dx > 0 && _dy > 0) {
BITMAPINFO bmi;
//memset( &bmi, 0, sizeof(bmi) );
bmi.bmiHeader.biSize = (bmi.bmiHeader.sizeof);
bmi.bmiHeader.biWidth = _dx;
bmi.bmiHeader.biHeight = _dy;
bmi.bmiHeader.biPlanes = 1;
bmi.bmiHeader.biBitCount = 32;
bmi.bmiHeader.biCompression = BI_RGB;
bmi.bmiHeader.biSizeImage = 0;
bmi.bmiHeader.biXPelsPerMeter = 1024;
bmi.bmiHeader.biYPelsPerMeter = 1024;
bmi.bmiHeader.biClrUsed = 0;
bmi.bmiHeader.biClrImportant = 0;
_drawbmp = CreateDIBSection( NULL, &bmi, DIB_RGB_COLORS, cast(void**)(&_pixels), NULL, 0 );
_drawdc = CreateCompatibleDC(NULL);
SelectObject(_drawdc, _drawbmp);
}
}
override void fill(uint color) {
int len = _dx * _dy;
for (int i = 0; i < len; i++)
_pixels[i] = color;
}
void drawTo(HDC dc, int x, int y) {
BitBlt(dc, x, y, _dx, _dx, _drawdc, 0, 0, SRCCOPY);
}
}
}

View File

@ -0,0 +1,479 @@
module dlangui.platforms.windows.win32fonts;
version (Windows) {
import win32.windows;
import dlangui.graphics.fonts;
import dlangui.platforms.windows.win32drawbuf;
import std.string;
import std.utf;
//auto toUTF16z(S)(S s)
//{
//return toUTFz!(const(wchar)*)(s);
//}
struct FontDef {
immutable FontFamily _family;
immutable string _face;
immutable ubyte _pitchAndFamily;
@property FontFamily family() { return _family; }
@property string face() { return _face; }
@property ubyte pitchAndFamily() { return _pitchAndFamily; }
this(FontFamily family, string face, ubyte putchAndFamily) {
_family = family;
_face = face;
_pitchAndFamily = pitchAndFamily;
}
}
/**
* Font implementation based on Win32 API system fonts.
*/
class Win32Font : Font {
HFONT _hfont;
int _size;
int _height;
int _weight;
int _baseline;
bool _italic;
string _face;
FontFamily _family;
LOGFONTA _logfont;
Win32ColorDrawBuf _drawbuf;
GlyphCache _glyphCache;
/// need to call create() after construction to initialize font
this() {
}
/// do cleanup
~this() {
clear();
}
/// cleanup resources
override void clear() {
if (_hfont !is null)
{
DeleteObject(_hfont);
_hfont = NULL;
_height = 0;
_baseline = 0;
_size = 0;
}
if (_drawbuf !is null) {
destroy(_drawbuf);
_drawbuf = null;
}
}
uint getGlyphIndex(dchar code)
{
if (_drawbuf is null)
return 0;
wchar[2] s;
wchar[2] g;
s[0] = cast(wchar)code;
s[1] = 0;
g[0] = 0;
GCP_RESULTSW gcp;
gcp.lStructSize = GCP_RESULTSW.sizeof;
gcp.lpOutString = null;
gcp.lpOrder = null;
gcp.lpDx = null;
gcp.lpCaretPos = null;
gcp.lpClass = null;
gcp.lpGlyphs = g.ptr;
gcp.nGlyphs = 2;
gcp.nMaxFit = 2;
DWORD res = GetCharacterPlacementW(
_drawbuf.dc, s.ptr, 1,
1000,
&gcp,
0
);
if (!res)
return 0;
return g[0];
}
override Glyph * getCharGlyph(dchar ch) {
uint glyphIndex = getGlyphIndex(ch);
if (!glyphIndex)
return null;
if (glyphIndex >= 0xFFFF)
return null;
Glyph * found = _glyphCache.find(cast(ushort)glyphIndex);
if (found !is null)
return found;
GLYPHMETRICS metrics;
MAT2 identity = { {0,1}, {0,0}, {0,0}, {0,1} };
uint res;
res = GetGlyphOutlineW( _drawbuf.dc, cast(wchar)ch,
GGO_METRICS,
&metrics,
0,
null,
&identity );
if (res==GDI_ERROR)
return null;
int gs = GetGlyphOutlineW( _drawbuf.dc, cast(wchar)ch,
GGO_GRAY8_BITMAP, //GGO_METRICS
&metrics,
0,
NULL,
&identity );
if (gs >= 0x10000 || gs < 0)
return null;
Glyph g;
g.blackBoxX = cast(ubyte)metrics.gmBlackBoxX;
g.blackBoxY = cast(ubyte)metrics.gmBlackBoxY;
g.originX = cast(byte)metrics.gmptGlyphOrigin.x;
g.originY = cast(byte)metrics.gmptGlyphOrigin.y;
g.width = cast(ubyte)metrics.gmCellIncX;
g.glyphIndex = cast(ushort)glyphIndex;
if (g.blackBoxX>0 && g.blackBoxY>0)
{
g.glyph = new ubyte[g.blackBoxX * g.blackBoxY];
if (gs>0)
{
ubyte glyph[] = new ubyte[gs];
res = GetGlyphOutlineW( _drawbuf.dc, cast(wchar)ch,
GGO_GRAY8_BITMAP, //GGO_METRICS
&metrics,
gs,
glyph.ptr,
&identity );
if (res==GDI_ERROR)
{
return null;
}
int glyph_row_size = (g.blackBoxX + 3) / 4 * 4;
ubyte * src = glyph.ptr;
ubyte * dst = g.glyph.ptr;
for (int y = 0; y < g.blackBoxY; y++)
{
for (int x = 0; x < g.blackBoxX; x++)
{
ubyte b = src[x];
if (b>=64)
b = 63;
b = (b<<2) & 0xFC;
dst[x] = b;
}
src += glyph_row_size;
dst += g.blackBoxX;
}
}
else
{
// empty glyph
for (int i = g.blackBoxX * g.blackBoxY - 1; i >= 0; i--)
g.glyph[i] = 0;
}
}
// found!
return _glyphCache.put(cast(ushort)glyphIndex, &g);
}
// draw text string to buffer
override void drawText(DrawBuf buf, int x, int y, const dchar[] text, uint color) {
int[] widths;
int charsMeasured = measureText(text, widths, 3000);
for (int i = 0; i < charsMeasured; i++) {
int xx = (i > 0) ? widths[i - 1] : 0;
Glyph * glyph = getCharGlyph(text[i]);
if (glyph is null)
continue;
if ( glyph.blackBoxX && glyph.blackBoxY ) {
buf.drawGlyph( x + xx + glyph.originX,
y + _baseline - glyph.originY,
glyph.glyph,
glyph.blackBoxX,
glyph.blackBoxY,
color);
}
}
}
override int measureText(const dchar[] text, ref int[] widths, int maxWidth) {
if (_hfont is null || _drawbuf is null || text.length == 0)
return 0;
wstring utf16text = toUTF16(text);
const wchar * pstr = utf16text.ptr;
uint len = cast(uint)utf16text.length;
GCP_RESULTSW gcpres;
gcpres.lStructSize = gcpres.sizeof;
if (widths.length < len + 1)
widths.length = len + 1;
gcpres.lpDx = widths.ptr;
gcpres.nMaxFit = len;
gcpres.nGlyphs = len;
uint res = GetCharacterPlacementW(
_drawbuf.dc,
pstr,
len,
maxWidth,
&gcpres,
GCP_MAXEXTENT); //|GCP_USEKERNING
if (!res) {
widths[0] = 0;
return 0;
}
uint measured = gcpres.nMaxFit;
int total = 0;
for (int i = 0; i < measured; i++) {
int w = widths[i];
total += w;
widths[i] = total;
}
return measured;
}
bool create(FontDef * def, int size, int weight, bool italic) {
if (!isNull())
clear();
LOGFONTA lf;
lf.lfCharSet = ANSI_CHARSET; //DEFAULT_CHARSET;
lf.lfFaceName[0..def.face.length] = def.face;
lf.lfFaceName[def.face.length] = 0;
lf.lfHeight = -size;
lf.lfItalic = italic;
lf.lfOutPrecision = OUT_TT_ONLY_PRECIS;
lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
lf.lfQuality = ANTIALIASED_QUALITY;
lf.lfPitchAndFamily = def.pitchAndFamily;
_hfont = CreateFontIndirectA(&lf);
_drawbuf = new Win32ColorDrawBuf(1, 1);
SelectObject(_drawbuf.dc, _hfont);
TEXTMETRICW tm;
GetTextMetricsW(_drawbuf.dc, &tm);
_size = size;
_height = tm.tmHeight;
_baseline = _height - tm.tmDescent;
_weight = weight;
_italic = italic;
_face = def.face;
_family = def.family;
Log.d("Created font ", _face, " ", _size);
return true;
}
// clear usage flags for all entries
override void checkpoint() {
_glyphCache.checkpoint();
}
// removes entries not used after last call of checkpoint() or cleanup()
override void cleanup() {
_glyphCache.cleanup();
}
@property override int size() { return _size; }
@property override int height() { return _height; }
@property override int weight() { return _weight; }
@property override int baseline() { return _baseline; }
@property override bool italic() { return _italic; }
@property override string face() { return _face; }
@property override FontFamily family() { return _family; }
@property override bool isNull() { return _hfont is null; }
}
/**
* Font manager implementation based on Win32 API system fonts.
*/
class Win32FontManager : FontManager {
FontList _activeFonts;
FontDef[] _fontFaces;
FontDef*[string] _faceByName;
/// initialize in constructor
this() {
instance = this;
init();
}
/// initialize font manager by enumerating of system fonts
bool init() {
Log.i("Win32FontManager.init()");
Win32ColorDrawBuf drawbuf = new Win32ColorDrawBuf(1,1);
LOGFONTA lf;
lf.lfCharSet = ANSI_CHARSET; //DEFAULT_CHARSET;
lf.lfFaceName[0] = 0;
HDC dc = drawbuf.dc;
int res =
EnumFontFamiliesExA(
dc, // handle to DC
&lf, // font information
&LVWin32FontEnumFontFamExProc, // callback function (FONTENUMPROC)
cast(LPARAM)(cast(void*)this), // additional data
0U // not used; must be 0
);
destroy(drawbuf);
Log.i("Found ", _fontFaces.length, " font faces");
return res!=0;
}
/// for returning of not found font
FontRef _emptyFontRef;
/// get font by properties
override ref FontRef getFont(int size, int weight, bool italic, FontFamily family, string face) {
Log.i("getFont()");
FontDef * def = findFace(family, face);
if (def !is null) {
int index = _activeFonts.find(size, weight, italic, def.family, def.face);
if (index >= 0)
return _activeFonts.get(index);
Log.d("Creating new font");
Win32Font item = new Win32Font();
if (!item.create(def, size, weight, italic))
return _emptyFontRef;
Log.d("Adding to list of active fonts");
return _activeFonts.add(item);
} else {
return _emptyFontRef;
}
}
/// find font face definition by family only (try to get one of defaults for family if possible)
FontDef * findFace(FontFamily family) {
FontDef * res = null;
switch(family) {
case FontFamily.SansSerif:
res = findFace("Arial"); if (res !is null) return res;
res = findFace("Tahoma"); if (res !is null) return res;
res = findFace("Calibri"); if (res !is null) return res;
res = findFace("Verdana"); if (res !is null) return res;
res = findFace("Lucida Sans"); if (res !is null) return res;
break;
case FontFamily.Serif:
res = findFace("Times New Roman"); if (res !is null) return res;
res = findFace("Georgia"); if (res !is null) return res;
res = findFace("Century Schoolbook"); if (res !is null) return res;
res = findFace("Bookman Old Style"); if (res !is null) return res;
break;
case FontFamily.MonoSpace:
res = findFace("Courier New"); if (res !is null) return res;
res = findFace("Lucida Console"); if (res !is null) return res;
res = findFace("Century Schoolbook"); if (res !is null) return res;
res = findFace("Bookman Old Style"); if (res !is null) return res;
break;
case FontFamily.Cursive:
res = findFace("Comic Sans MS"); if (res !is null) return res;
res = findFace("Lucida Handwriting"); if (res !is null) return res;
res = findFace("Monotype Corsiva"); if (res !is null) return res;
break;
default:
break;
}
return null;
}
/// find font face definition by face only
FontDef * findFace(string face) {
if (face.length == 0)
return null;
if (face in _faceByName)
return _faceByName[face];
return null;
}
/// find font face definition by family and face
FontDef * findFace(FontFamily family, string face) {
// by face only
FontDef * res = findFace(face);
if (res !is null)
return res;
// best for family
res = findFace(family);
if (res !is null)
return res;
for (int i = 0; i < _fontFaces.length; i++) {
res = &_fontFaces[i];
if (res.family == family)
return res;
}
res = findFace(FontFamily.SansSerif);
if (res !is null)
return res;
return &_fontFaces[0];
}
/// register enumerated font
bool registerFont(FontFamily family, string fontFace, ubyte pitchAndFamily) {
Log.d("registerFont(", family, ",", fontFace, ")");
_fontFaces ~= FontDef(family, fontFace, pitchAndFamily);
_faceByName[fontFace] = &_fontFaces[$ - 1];
return true;
}
/// clear usage flags for all entries
override void checkpoint() {
_activeFonts.checkpoint();
}
/// removes entries not used after last call of checkpoint() or cleanup()
override void cleanup() {
_activeFonts.cleanup();
//_list.cleanup();
}
}
string fromStringz(const(char[]) s) {
int i = 0;
while(s[i])
i++;
return cast(string)(s[0..i].dup);
}
FontFamily pitchAndFamilyToFontFamily(ubyte flags) {
if ((flags & FF_DECORATIVE) == FF_DECORATIVE)
return FontFamily.Fantasy;
else if ((flags & (FIXED_PITCH)) != 0) // | | MONO_FONT
return FontFamily.MonoSpace;
else if ((flags & (FF_ROMAN)) != 0)
return FontFamily.Serif;
else if ((flags & (FF_SCRIPT)) != 0)
return FontFamily.Cursive;
return FontFamily.SansSerif;
}
// definition
extern(Windows) {
int LVWin32FontEnumFontFamExProc(
const (LOGFONTA) *lf, // logical-font data
const (TEXTMETRICA) *lpntme, // physical-font data
//ENUMLOGFONTEX *lpelfe, // logical-font data
//NEWTEXTMETRICEX *lpntme, // physical-font data
DWORD fontType, // type of font
LPARAM lParam // application-defined data
)
{
//
//Log.d("LVWin32FontEnumFontFamExProc fontType=", fontType);
if (fontType == TRUETYPE_FONTTYPE)
{
void * p = cast(void*)lParam;
Win32FontManager fontman = cast(Win32FontManager)p;
string face = fromStringz(lf.lfFaceName);
FontFamily family = pitchAndFamilyToFontFamily(lf.lfPitchAndFamily);
if (face.length < 2 || face[0] == '@')
return 1;
//Log.d("face:", face);
fontman.registerFont(family, face, lf.lfPitchAndFamily);
}
return 1;
}
}
}

View File

@ -9,6 +9,8 @@ import std.utf;
import std.stdio;
import std.algorithm;
import dlangui.platforms.common.platform;
import dlangui.platforms.windows.win32fonts;
import dlangui.platforms.windows.win32drawbuf;
import dlangui.graphics.drawbuf;
import dlangui.graphics.fonts;
import dlangui.core.logger;
@ -17,468 +19,6 @@ pragma(lib, "gdi32.lib");
pragma(lib, "user32.lib");
pragma(lib, "libpng15.lib");
struct FontDef {
immutable FontFamily _family;
immutable string _face;
immutable ubyte _pitchAndFamily;
@property FontFamily family() { return _family; }
@property string face() { return _face; }
@property ubyte pitchAndFamily() { return _pitchAndFamily; }
this(FontFamily family, string face, ubyte putchAndFamily) {
_family = family;
_face = face;
_pitchAndFamily = pitchAndFamily;
}
}
/**
* Font implementation based on Win32 API system fonts.
*/
class Win32Font : Font {
HFONT _hfont;
int _size;
int _height;
int _weight;
int _baseline;
bool _italic;
string _face;
FontFamily _family;
LOGFONTA _logfont;
Win32ColorDrawBuf _drawbuf;
GlyphCache _glyphCache;
/// need to call create() after construction to initialize font
this() {
}
/// do cleanup
~this() {
clear();
}
/// cleanup resources
override void clear() {
if (_hfont !is null)
{
DeleteObject(_hfont);
_hfont = NULL;
_height = 0;
_baseline = 0;
_size = 0;
}
if (_drawbuf !is null) {
destroy(_drawbuf);
_drawbuf = null;
}
}
uint getGlyphIndex(dchar code)
{
if (_drawbuf is null)
return 0;
wchar[2] s;
wchar[2] g;
s[0] = cast(wchar)code;
s[1] = 0;
g[0] = 0;
GCP_RESULTSW gcp;
gcp.lStructSize = GCP_RESULTSW.sizeof;
gcp.lpOutString = null;
gcp.lpOrder = null;
gcp.lpDx = null;
gcp.lpCaretPos = null;
gcp.lpClass = null;
gcp.lpGlyphs = g.ptr;
gcp.nGlyphs = 2;
gcp.nMaxFit = 2;
DWORD res = GetCharacterPlacementW(
_drawbuf.dc, s.ptr, 1,
1000,
&gcp,
0
);
if (!res)
return 0;
return g[0];
}
override Glyph * getCharGlyph(dchar ch) {
uint glyphIndex = getGlyphIndex(ch);
if (!glyphIndex)
return null;
if (glyphIndex >= 0xFFFF)
return null;
Glyph * found = _glyphCache.find(cast(ushort)glyphIndex);
if (found !is null)
return found;
GLYPHMETRICS metrics;
MAT2 identity = { {0,1}, {0,0}, {0,0}, {0,1} };
uint res;
res = GetGlyphOutlineW( _drawbuf.dc, cast(wchar)ch,
GGO_METRICS,
&metrics,
0,
null,
&identity );
if (res==GDI_ERROR)
return null;
int gs = GetGlyphOutlineW( _drawbuf.dc, cast(wchar)ch,
GGO_GRAY8_BITMAP, //GGO_METRICS
&metrics,
0,
NULL,
&identity );
if (gs >= 0x10000 || gs < 0)
return null;
Glyph g;
g.blackBoxX = cast(ubyte)metrics.gmBlackBoxX;
g.blackBoxY = cast(ubyte)metrics.gmBlackBoxY;
g.originX = cast(byte)metrics.gmptGlyphOrigin.x;
g.originY = cast(byte)metrics.gmptGlyphOrigin.y;
g.width = cast(ubyte)metrics.gmCellIncX;
g.glyphIndex = cast(ushort)glyphIndex;
if (g.blackBoxX>0 && g.blackBoxY>0)
{
g.glyph = new ubyte[g.blackBoxX * g.blackBoxY];
if (gs>0)
{
ubyte glyph[] = new ubyte[gs];
res = GetGlyphOutlineW( _drawbuf.dc, cast(wchar)ch,
GGO_GRAY8_BITMAP, //GGO_METRICS
&metrics,
gs,
glyph.ptr,
&identity );
if (res==GDI_ERROR)
{
return null;
}
int glyph_row_size = (g.blackBoxX + 3) / 4 * 4;
ubyte * src = glyph.ptr;
ubyte * dst = g.glyph.ptr;
for (int y = 0; y < g.blackBoxY; y++)
{
for (int x = 0; x < g.blackBoxX; x++)
{
ubyte b = src[x];
if (b>=64)
b = 63;
b = (b<<2) & 0xFC;
dst[x] = b;
}
src += glyph_row_size;
dst += g.blackBoxX;
}
}
else
{
// empty glyph
for (int i = g.blackBoxX * g.blackBoxY - 1; i >= 0; i--)
g.glyph[i] = 0;
}
}
// found!
return _glyphCache.put(cast(ushort)glyphIndex, &g);
}
// draw text string to buffer
override void drawText(DrawBuf buf, int x, int y, const dchar[] text, uint color) {
int[] widths;
int charsMeasured = measureText(text, widths, 3000);
for (int i = 0; i < charsMeasured; i++) {
int xx = (i > 0) ? widths[i - 1] : 0;
Glyph * glyph = getCharGlyph(text[i]);
if (glyph is null)
continue;
if ( glyph.blackBoxX && glyph.blackBoxY ) {
buf.drawGlyph( x + xx + glyph.originX,
y + _baseline - glyph.originY,
glyph.glyph,
glyph.blackBoxX,
glyph.blackBoxY,
color);
}
}
}
override int measureText(const dchar[] text, ref int[] widths, int maxWidth) {
if (_hfont is null || _drawbuf is null || text.length == 0)
return 0;
wstring utf16text = toUTF16(text);
const wchar * pstr = utf16text.ptr;
uint len = cast(uint)utf16text.length;
GCP_RESULTSW gcpres;
gcpres.lStructSize = gcpres.sizeof;
if (widths.length < len + 1)
widths.length = len + 1;
gcpres.lpDx = widths.ptr;
gcpres.nMaxFit = len;
gcpres.nGlyphs = len;
uint res = GetCharacterPlacementW(
_drawbuf.dc,
pstr,
len,
maxWidth,
&gcpres,
GCP_MAXEXTENT); //|GCP_USEKERNING
if (!res) {
widths[0] = 0;
return 0;
}
uint measured = gcpres.nMaxFit;
int total = 0;
for (int i = 0; i < measured; i++) {
int w = widths[i];
total += w;
widths[i] = total;
}
return measured;
}
bool create(FontDef * def, int size, int weight, bool italic) {
if (!isNull())
clear();
LOGFONTA lf;
lf.lfCharSet = ANSI_CHARSET; //DEFAULT_CHARSET;
lf.lfFaceName[0..def.face.length] = def.face;
lf.lfFaceName[def.face.length] = 0;
lf.lfHeight = -size;
lf.lfItalic = italic;
lf.lfOutPrecision = OUT_TT_ONLY_PRECIS;
lf.lfClipPrecision = CLIP_DEFAULT_PRECIS;
lf.lfQuality = ANTIALIASED_QUALITY;
lf.lfPitchAndFamily = def.pitchAndFamily;
_hfont = CreateFontIndirectA(&lf);
_drawbuf = new Win32ColorDrawBuf(1, 1);
SelectObject(_drawbuf.dc, _hfont);
TEXTMETRICW tm;
GetTextMetricsW(_drawbuf.dc, &tm);
_size = size;
_height = tm.tmHeight;
_baseline = _height - tm.tmDescent;
_weight = weight;
_italic = italic;
_face = def.face;
_family = def.family;
Log.d("Created font ", _face, " ", _size);
return true;
}
// clear usage flags for all entries
override void checkpoint() {
_glyphCache.checkpoint();
}
// removes entries not used after last call of checkpoint() or cleanup()
override void cleanup() {
_glyphCache.cleanup();
}
@property override int size() { return _size; }
@property override int height() { return _height; }
@property override int weight() { return _weight; }
@property override int baseline() { return _baseline; }
@property override bool italic() { return _italic; }
@property override string face() { return _face; }
@property override FontFamily family() { return _family; }
@property override bool isNull() { return _hfont is null; }
}
/**
* Font manager implementation based on Win32 API system fonts.
*/
class Win32FontManager : FontManager {
FontList _activeFonts;
FontDef[] _fontFaces;
FontDef*[string] _faceByName;
/// initialize in constructor
this() {
instance = this;
init();
}
/// initialize font manager by enumerating of system fonts
bool init() {
Log.i("Win32FontManager.init()");
Win32ColorDrawBuf drawbuf = new Win32ColorDrawBuf(1,1);
LOGFONTA lf;
lf.lfCharSet = ANSI_CHARSET; //DEFAULT_CHARSET;
lf.lfFaceName[0] = 0;
HDC dc = drawbuf.dc;
int res =
EnumFontFamiliesExA(
dc, // handle to DC
&lf, // font information
&LVWin32FontEnumFontFamExProc, // callback function (FONTENUMPROC)
cast(LPARAM)(cast(void*)this), // additional data
0U // not used; must be 0
);
destroy(drawbuf);
Log.i("Found ", _fontFaces.length, " font faces");
return res!=0;
}
/// for returning of not found font
FontRef _emptyFontRef;
/// get font by properties
override ref FontRef getFont(int size, int weight, bool italic, FontFamily family, string face) {
Log.i("getFont()");
FontDef * def = findFace(family, face);
if (def !is null) {
int index = _activeFonts.find(size, weight, italic, def.family, def.face);
if (index >= 0)
return _activeFonts.get(index);
Log.d("Creating new font");
Win32Font item = new Win32Font();
if (!item.create(def, size, weight, italic))
return _emptyFontRef;
Log.d("Adding to list of active fonts");
return _activeFonts.add(item);
} else {
return _emptyFontRef;
}
}
/// find font face definition by family only (try to get one of defaults for family if possible)
FontDef * findFace(FontFamily family) {
FontDef * res = null;
switch(family) {
case FontFamily.SansSerif:
res = findFace("Arial"); if (res !is null) return res;
res = findFace("Tahoma"); if (res !is null) return res;
res = findFace("Calibri"); if (res !is null) return res;
res = findFace("Verdana"); if (res !is null) return res;
res = findFace("Lucida Sans"); if (res !is null) return res;
break;
case FontFamily.Serif:
res = findFace("Times New Roman"); if (res !is null) return res;
res = findFace("Georgia"); if (res !is null) return res;
res = findFace("Century Schoolbook"); if (res !is null) return res;
res = findFace("Bookman Old Style"); if (res !is null) return res;
break;
case FontFamily.MonoSpace:
res = findFace("Courier New"); if (res !is null) return res;
res = findFace("Lucida Console"); if (res !is null) return res;
res = findFace("Century Schoolbook"); if (res !is null) return res;
res = findFace("Bookman Old Style"); if (res !is null) return res;
break;
case FontFamily.Cursive:
res = findFace("Comic Sans MS"); if (res !is null) return res;
res = findFace("Lucida Handwriting"); if (res !is null) return res;
res = findFace("Monotype Corsiva"); if (res !is null) return res;
break;
default:
break;
}
return null;
}
/// find font face definition by face only
FontDef * findFace(string face) {
if (face.length == 0)
return null;
if (face in _faceByName)
return _faceByName[face];
return null;
}
/// find font face definition by family and face
FontDef * findFace(FontFamily family, string face) {
// by face only
FontDef * res = findFace(face);
if (res !is null)
return res;
// best for family
res = findFace(family);
if (res !is null)
return res;
for (int i = 0; i < _fontFaces.length; i++) {
res = &_fontFaces[i];
if (res.family == family)
return res;
}
res = findFace(FontFamily.SansSerif);
if (res !is null)
return res;
return &_fontFaces[0];
}
/// register enumerated font
bool registerFont(FontFamily family, string fontFace, ubyte pitchAndFamily) {
Log.d("registerFont(", family, ",", fontFace, ")");
_fontFaces ~= FontDef(family, fontFace, pitchAndFamily);
_faceByName[fontFace] = &_fontFaces[$ - 1];
return true;
}
/// clear usage flags for all entries
override void checkpoint() {
_activeFonts.checkpoint();
}
/// removes entries not used after last call of checkpoint() or cleanup()
override void cleanup() {
_activeFonts.cleanup();
//_list.cleanup();
}
}
string fromStringz(const(char[]) s) {
int i = 0;
while(s[i])
i++;
return cast(string)(s[0..i].dup);
}
FontFamily pitchAndFamilyToFontFamily(ubyte flags) {
if ((flags & FF_DECORATIVE) == FF_DECORATIVE)
return FontFamily.Fantasy;
else if ((flags & (FIXED_PITCH)) != 0) // | | MONO_FONT
return FontFamily.MonoSpace;
else if ((flags & (FF_ROMAN)) != 0)
return FontFamily.Serif;
else if ((flags & (FF_SCRIPT)) != 0)
return FontFamily.Cursive;
return FontFamily.SansSerif;
}
// definition
extern(Windows) {
int LVWin32FontEnumFontFamExProc(
const (LOGFONTA) *lf, // logical-font data
const (TEXTMETRICA) *lpntme, // physical-font data
//ENUMLOGFONTEX *lpelfe, // logical-font data
//NEWTEXTMETRICEX *lpntme, // physical-font data
DWORD fontType, // type of font
LPARAM lParam // application-defined data
)
{
//
//Log.d("LVWin32FontEnumFontFamExProc fontType=", fontType);
if (fontType == TRUETYPE_FONTTYPE)
{
void * p = cast(void*)lParam;
Win32FontManager fontman = cast(Win32FontManager)p;
string face = fromStringz(lf.lfFaceName);
FontFamily family = pitchAndFamilyToFontFamily(lf.lfPitchAndFamily);
if (face.length < 2 || face[0] == '@')
return 1;
//Log.d("face:", face);
fontman.registerFont(family, face, lf.lfPitchAndFamily);
}
return 1;
}
}
extern (C) int UIAppMain();
immutable WIN_CLASS_NAME = "DLANGUI_APP";
@ -572,11 +112,6 @@ class Win32Platform : Platform {
}
}
auto toUTF16z(S)(S s)
{
return toUTFz!(const(wchar)*)(s);
}
extern (Windows)
int WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
LPSTR lpCmdLine, int nCmdShow)
@ -625,103 +160,6 @@ int myWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int
return UIAppMain();
}
class Win32ColorDrawBuf : ColorDrawBufBase {
uint * _pixels;
HDC _drawdc;
HBITMAP _drawbmp;
@property HDC dc() { return _drawdc; }
this(int width, int height) {
resize(width, height);
}
override uint * scanLine(int y) {
if (y >= 0 && y < _dy)
return _pixels + _dx * (_dy - 1 - y);
return null;
}
override void clear() {
if (_drawbmp !is null) {
DeleteObject( _drawbmp );
DeleteObject( _drawdc );
_pixels = null;
_dx = 0;
_dy = 0;
}
}
override void resize(int width, int height) {
if (width< 0)
width = 0;
if (height < 0)
height = 0;
if (_dx == width && _dy == height)
return;
clear();
_dx = width;
_dy = height;
if (_dx > 0 && _dy > 0) {
BITMAPINFO bmi;
//memset( &bmi, 0, sizeof(bmi) );
bmi.bmiHeader.biSize = (bmi.bmiHeader.sizeof);
bmi.bmiHeader.biWidth = _dx;
bmi.bmiHeader.biHeight = _dy;
bmi.bmiHeader.biPlanes = 1;
bmi.bmiHeader.biBitCount = 32;
bmi.bmiHeader.biCompression = BI_RGB;
bmi.bmiHeader.biSizeImage = 0;
bmi.bmiHeader.biXPelsPerMeter = 1024;
bmi.bmiHeader.biYPelsPerMeter = 1024;
bmi.bmiHeader.biClrUsed = 0;
bmi.bmiHeader.biClrImportant = 0;
_drawbmp = CreateDIBSection( NULL, &bmi, DIB_RGB_COLORS, cast(void**)(&_pixels), NULL, 0 );
_drawdc = CreateCompatibleDC(NULL);
SelectObject(_drawdc, _drawbmp);
}
}
override void fill(uint color) {
int len = _dx * _dy;
for (int i = 0; i < len; i++)
_pixels[i] = color;
}
void drawTo(HDC dc, int x, int y) {
BitBlt(dc, x, y, _dx, _dx, _drawdc, 0, 0, SRCCOPY);
}
}
//void drawBuf2DC(HDC dc, int x, int y, DrawBuf buf)
//{
// uint * drawpixels;
// HDC drawdc;
// HBITMAP drawbmp;
//
// int buf_width = buf.width();
// int bytesPerRow = buf_width * 4;
// BITMAPINFO bmi;
// //memset( &bmi, 0, sizeof(bmi) );
// bmi.bmiHeader.biSize = (bmi.bmiHeader.sizeof);
// bmi.bmiHeader.biWidth = buf_width;
// bmi.bmiHeader.biHeight = buf.height;
// bmi.bmiHeader.biPlanes = 1;
// bmi.bmiHeader.biBitCount = 32;
// bmi.bmiHeader.biCompression = BI_RGB;
// bmi.bmiHeader.biSizeImage = 0;
// bmi.bmiHeader.biXPelsPerMeter = 1024;
// bmi.bmiHeader.biYPelsPerMeter = 1024;
// bmi.bmiHeader.biClrUsed = 0;
// bmi.bmiHeader.biClrImportant = 0;
// drawbmp = CreateDIBSection( NULL, &bmi, DIB_RGB_COLORS, cast(void**)(&drawpixels), NULL, 0 );
// drawdc = CreateCompatibleDC(NULL);
// SelectObject(drawdc, drawbmp);
// for (int yy=0; yy < buf.height; yy++)
// {
// uint * src = buf.scanLine(yy);
// uint * dst = drawpixels + (buf.height - 1 - yy) * buf.width;
// for (int xx = 0; xx < buf_width; xx++)
// dst[xx] = src[xx];
// }
// BitBlt( dc, x, y, buf_width, buf.height, drawdc, 0, 0, SRCCOPY);
// DeleteObject( drawbmp );
// DeleteObject( drawdc );
//}
extern(Windows)
LRESULT WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)