2016-04-14 20:35:50 -05:00
// 14 april 2016
2016-04-23 15:49:46 -05:00
# include "uipriv_windows.hpp"
2018-03-17 13:47:20 -05:00
# include "attrstr.hpp"
2016-04-14 20:35:50 -05:00
2016-04-17 18:37:03 -05:00
// TODOs
2016-04-17 21:09:18 -05:00
// - quote the Choose Font sample here for reference
2016-04-17 18:37:03 -05:00
// - the Choose Font sample defaults to Regular/Italic/Bold/Bold Italic in some case (no styles?); do we? find out what the case is
2016-04-17 21:09:18 -05:00
// - do we set initial family and style topmost as well?
2016-04-18 12:32:48 -05:00
// - this should probably just handle IDWriteFonts
2017-02-22 10:23:26 -06:00
// - localization?
// - the Sample window overlaps the groupbox in a weird way (compare to the real ChooseFont() dialog)
2016-04-17 18:37:03 -05:00
2016-04-14 20:35:50 -05:00
struct fontDialog {
HWND hwnd ;
HWND familyCombobox ;
HWND styleCombobox ;
HWND sizeCombobox ;
2016-04-17 00:34:05 -05:00
2016-04-18 12:32:48 -05:00
struct fontDialogParams * params ;
2016-04-17 00:34:05 -05:00
2018-03-17 14:49:00 -05:00
struct fontCollection * fc ;
2016-04-17 00:34:05 -05:00
2016-04-16 16:58:45 -05:00
RECT sampleRect ;
2016-04-17 12:06:45 -05:00
HWND sampleBox ;
2016-04-17 00:34:05 -05:00
// we store the current selections in case an invalid string is typed in (partial or nonexistent or invalid number)
// on OK, these are what are read
LRESULT curFamily ;
LRESULT curStyle ;
2016-04-17 18:03:34 -05:00
double curSize ;
2016-04-17 18:22:37 -05:00
2016-04-17 18:37:03 -05:00
// these are finding the style that's closest to the previous one (these fields) when changing a font
2016-04-17 18:22:37 -05:00
DWRITE_FONT_WEIGHT weight ;
DWRITE_FONT_STYLE style ;
DWRITE_FONT_STRETCH stretch ;
2016-04-14 20:35:50 -05:00
} ;
2016-04-17 19:10:14 -05:00
static LRESULT cbAddString ( HWND cb , const WCHAR * str )
2016-04-14 21:25:32 -05:00
{
LRESULT lr ;
lr = SendMessageW ( cb , CB_ADDSTRING , 0 , ( LPARAM ) str ) ;
if ( lr = = ( LRESULT ) CB_ERR | | lr = = ( LRESULT ) CB_ERRSPACE )
2016-04-23 21:15:33 -05:00
logLastError ( L " error adding item to combobox " ) ;
2016-04-14 21:25:32 -05:00
return lr ;
}
2016-04-17 19:10:14 -05:00
static LRESULT cbInsertString ( HWND cb , const WCHAR * str , WPARAM pos )
2016-04-14 21:25:32 -05:00
{
LRESULT lr ;
2016-04-17 19:10:14 -05:00
lr = SendMessageW ( cb , CB_INSERTSTRING , pos , ( LPARAM ) str ) ;
if ( lr ! = ( LRESULT ) pos )
2016-04-23 21:15:33 -05:00
logLastError ( L " error inserting item to combobox " ) ;
2016-04-14 21:25:32 -05:00
return lr ;
}
2016-04-17 00:34:05 -05:00
static LRESULT cbGetItemData ( HWND cb , WPARAM item )
2016-04-15 12:18:06 -05:00
{
2016-04-17 00:34:05 -05:00
LRESULT data ;
data = SendMessageW ( cb , CB_GETITEMDATA , item , 0 ) ;
if ( data = = ( LRESULT ) CB_ERR )
2016-04-23 21:15:33 -05:00
logLastError ( L " error getting combobox item data for font dialog " ) ;
2016-04-17 00:34:05 -05:00
return data ;
}
static void cbSetItemData ( HWND cb , WPARAM item , LPARAM data )
{
if ( SendMessageW ( cb , CB_SETITEMDATA , item , data ) = = ( LRESULT ) CB_ERR )
2016-04-23 21:15:33 -05:00
logLastError ( L " error setting combobox item data " ) ;
2016-04-17 00:34:05 -05:00
}
static BOOL cbGetCurSel ( HWND cb , LRESULT * sel )
{
LRESULT n ;
n = SendMessageW ( cb , CB_GETCURSEL , 0 , 0 ) ;
if ( n = = ( LRESULT ) CB_ERR )
return FALSE ;
if ( sel ! = NULL )
* sel = n ;
return TRUE ;
}
static void cbSetCurSel ( HWND cb , WPARAM item )
{
2016-04-17 18:03:34 -05:00
if ( SendMessageW ( cb , CB_SETCURSEL , item , 0 ) ! = ( LRESULT ) item )
2016-04-23 21:15:33 -05:00
logLastError ( L " error selecting combobox item " ) ;
2016-04-17 00:34:05 -05:00
}
2016-04-15 12:18:06 -05:00
2016-04-17 00:34:05 -05:00
static LRESULT cbGetCount ( HWND cb )
{
LRESULT n ;
n = SendMessageW ( cb , CB_GETCOUNT , 0 , 0 ) ;
2016-04-15 12:18:06 -05:00
if ( n = = ( LRESULT ) CB_ERR )
2016-04-23 21:15:33 -05:00
logLastError ( L " error getting combobox item count " ) ;
2016-04-17 00:34:05 -05:00
return n ;
}
static void cbWipeAndReleaseData ( HWND cb )
{
IUnknown * obj ;
LRESULT i , n ;
n = cbGetCount ( cb ) ;
2016-04-15 12:18:06 -05:00
for ( i = 0 ; i < n ; i + + ) {
2016-04-17 00:34:05 -05:00
obj = ( IUnknown * ) cbGetItemData ( cb , ( WPARAM ) i ) ;
obj - > Release ( ) ;
2016-04-15 12:18:06 -05:00
}
2016-04-17 00:34:05 -05:00
SendMessageW ( cb , CB_RESETCONTENT , 0 , 0 ) ;
}
2016-04-17 18:03:34 -05:00
static WCHAR * cbGetItemText ( HWND cb , WPARAM item )
{
LRESULT len ;
WCHAR * text ;
// note: neither message includes the terminating L'\0'
len = SendMessageW ( cb , CB_GETLBTEXTLEN , item , 0 ) ;
if ( len = = ( LRESULT ) CB_ERR )
2016-04-23 21:15:33 -05:00
logLastError ( L " error getting item text length from combobox " ) ;
2018-04-15 17:12:58 -05:00
text = ( WCHAR * ) uiprivAlloc ( ( len + 1 ) * sizeof ( WCHAR ) , " WCHAR[] " ) ;
2016-04-17 18:03:34 -05:00
if ( SendMessageW ( cb , CB_GETLBTEXT , item , ( LPARAM ) text ) ! = len )
2016-04-23 21:15:33 -05:00
logLastError ( L " error getting item text from combobox " ) ;
2016-04-17 19:10:14 -05:00
return text ;
2016-04-17 18:03:34 -05:00
}
static BOOL cbTypeToSelect ( HWND cb , LRESULT * posOut , BOOL restoreAfter )
{
WCHAR * text ;
LRESULT pos ;
DWORD selStart , selEnd ;
// start by saving the current selection as setting the item will change the selection
SendMessageW ( cb , CB_GETEDITSEL , ( WPARAM ) ( & selStart ) , ( LPARAM ) ( & selEnd ) ) ;
text = windowText ( cb ) ;
pos = SendMessageW ( cb , CB_FINDSTRINGEXACT , ( WPARAM ) ( - 1 ) , ( LPARAM ) text ) ;
if ( pos = = ( LRESULT ) CB_ERR ) {
2018-04-15 17:12:58 -05:00
uiprivFree ( text ) ;
2016-04-17 18:03:34 -05:00
return FALSE ;
}
cbSetCurSel ( cb , ( WPARAM ) pos ) ;
if ( posOut ! = NULL )
* posOut = pos ;
if ( restoreAfter )
if ( SendMessageW ( cb , WM_SETTEXT , 0 , ( LPARAM ) text ) ! = ( LRESULT ) TRUE )
2016-04-23 21:15:33 -05:00
logLastError ( L " error restoring old combobox text " ) ;
2018-04-15 17:12:58 -05:00
uiprivFree ( text ) ;
2016-04-17 18:03:34 -05:00
// and restore the selection like above
// TODO isn't there a 32-bit version of this
if ( SendMessageW ( cb , CB_SETEDITSEL , 0 , MAKELPARAM ( selStart , selEnd ) ) ! = ( LRESULT ) TRUE )
2016-04-23 21:15:33 -05:00
logLastError ( L " error restoring combobox edit selection " ) ;
2016-04-17 18:03:34 -05:00
return TRUE ;
}
2016-04-17 00:34:05 -05:00
static void wipeStylesBox ( struct fontDialog * f )
{
cbWipeAndReleaseData ( f - > styleCombobox ) ;
2016-04-15 12:18:06 -05:00
}
2016-04-18 12:32:48 -05:00
static WCHAR * fontStyleName ( struct fontCollection * fc , IDWriteFont * font )
2016-04-15 12:18:06 -05:00
{
IDWriteLocalizedStrings * str ;
WCHAR * wstr ;
HRESULT hr ;
2016-04-17 00:34:05 -05:00
hr = font - > GetFaceNames ( & str ) ;
2016-04-15 12:18:06 -05:00
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error getting font style name for font dialog " , hr ) ;
2018-03-17 14:49:00 -05:00
wstr = uiprivFontCollectionCorrectString ( fc , str ) ;
2016-04-15 12:18:06 -05:00
str - > Release ( ) ;
return wstr ;
}
2016-04-17 18:22:37 -05:00
static void queueRedrawSampleText ( struct fontDialog * f )
{
// TODO TRUE?
2016-05-17 11:41:41 -05:00
invalidateRect ( f - > sampleBox , NULL , TRUE ) ;
2016-04-17 18:22:37 -05:00
}
static void styleChanged ( struct fontDialog * f )
{
LRESULT pos ;
BOOL selected ;
IDWriteFont * font ;
selected = cbGetCurSel ( f - > styleCombobox , & pos ) ;
if ( ! selected ) // on deselect, do nothing
return ;
f - > curStyle = pos ;
font = ( IDWriteFont * ) cbGetItemData ( f - > styleCombobox , ( WPARAM ) ( f - > curStyle ) ) ;
// these are for the nearest match when changing the family; see below
f - > weight = font - > GetWeight ( ) ;
f - > style = font - > GetStyle ( ) ;
f - > stretch = font - > GetStretch ( ) ;
queueRedrawSampleText ( f ) ;
}
static void styleEdited ( struct fontDialog * f )
{
if ( cbTypeToSelect ( f - > styleCombobox , & ( f - > curStyle ) , FALSE ) )
styleChanged ( f ) ;
}
2016-04-15 12:18:06 -05:00
static void familyChanged ( struct fontDialog * f )
{
2016-04-17 00:34:05 -05:00
LRESULT pos ;
BOOL selected ;
2016-04-16 21:46:39 -05:00
IDWriteFontFamily * family ;
2016-04-17 18:37:03 -05:00
IDWriteFont * font , * matchFont ;
DWRITE_FONT_WEIGHT weight ;
DWRITE_FONT_STYLE style ;
DWRITE_FONT_STRETCH stretch ;
2016-04-16 21:46:39 -05:00
UINT32 i , n ;
2016-04-17 18:37:03 -05:00
UINT32 matching ;
2016-04-15 12:18:06 -05:00
WCHAR * label ;
HRESULT hr ;
2016-04-17 00:34:05 -05:00
selected = cbGetCurSel ( f - > familyCombobox , & pos ) ;
if ( ! selected ) // on deselect, do nothing
return ;
f - > curFamily = pos ;
2016-04-15 12:18:06 -05:00
2016-04-17 00:34:05 -05:00
family = ( IDWriteFontFamily * ) cbGetItemData ( f - > familyCombobox , ( WPARAM ) ( f - > curFamily ) ) ;
2016-04-15 12:18:06 -05:00
2016-04-17 18:37:03 -05:00
// for the nearest style match
// when we select a new family, we want the nearest style to the previously selected one to be chosen
// this is how the Choose Font sample does it
hr = family - > GetFirstMatchingFont (
f - > weight ,
f - > stretch ,
f - > style ,
& matchFont ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error finding first matching font to previous style in font dialog " , hr ) ;
2016-04-17 18:37:03 -05:00
// we can't just compare pointers; a "newly created" object comes out
// the Choose Font sample appears to do this instead
weight = matchFont - > GetWeight ( ) ;
style = matchFont - > GetStyle ( ) ;
stretch = matchFont - > GetStretch ( ) ;
matchFont - > Release ( ) ;
2016-04-16 21:46:39 -05:00
// TODO test mutliple streteches; all the fonts I have have only one stretch value?
2016-04-17 00:34:05 -05:00
wipeStylesBox ( f ) ;
2016-04-16 21:46:39 -05:00
n = family - > GetFontCount ( ) ;
2016-04-17 18:37:03 -05:00
matching = 0 ; // a safe/suitable default just in case
2016-04-16 21:46:39 -05:00
for ( i = 0 ; i < n ; i + + ) {
2016-04-17 00:34:05 -05:00
hr = family - > GetFont ( i , & font ) ;
2016-04-15 12:18:06 -05:00
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error getting font for filling styles box " , hr ) ;
2016-04-18 12:32:48 -05:00
label = fontStyleName ( f - > fc , font ) ;
2016-04-15 12:18:06 -05:00
pos = cbAddString ( f - > styleCombobox , label ) ;
2018-04-15 17:12:58 -05:00
uiprivFree ( label ) ;
2016-04-17 00:34:05 -05:00
cbSetItemData ( f - > styleCombobox , ( WPARAM ) pos , ( LPARAM ) font ) ;
2016-04-17 18:37:03 -05:00
if ( font - > GetWeight ( ) = = weight & &
font - > GetStyle ( ) = = style & &
font - > GetStretch ( ) = = stretch )
matching = i ;
2016-04-15 12:18:06 -05:00
}
2016-04-17 18:37:03 -05:00
// and now, load the match
cbSetCurSel ( f - > styleCombobox , ( WPARAM ) matching ) ;
2016-04-17 18:22:37 -05:00
styleChanged ( f ) ;
2016-04-17 12:06:45 -05:00
}
2016-04-17 18:03:34 -05:00
// TODO search language variants like the sample does
static void familyEdited ( struct fontDialog * f )
{
if ( cbTypeToSelect ( f - > familyCombobox , & ( f - > curFamily ) , FALSE ) )
familyChanged ( f ) ;
}
2016-04-17 12:06:45 -05:00
2016-04-17 19:10:14 -05:00
static const struct {
const WCHAR * text ;
double value ;
} defaultSizes [ ] = {
{ L " 8 " , 8 } ,
{ L " 9 " , 9 } ,
{ L " 10 " , 10 } ,
{ L " 11 " , 11 } ,
{ L " 12 " , 12 } ,
{ L " 14 " , 14 } ,
{ L " 16 " , 16 } ,
{ L " 18 " , 18 } ,
{ L " 20 " , 20 } ,
{ L " 22 " , 22 } ,
{ L " 24 " , 24 } ,
{ L " 26 " , 26 } ,
{ L " 28 " , 28 } ,
{ L " 36 " , 36 } ,
{ L " 48 " , 48 } ,
{ L " 72 " , 72 } ,
{ NULL , 0 } ,
} ;
static void sizeChanged ( struct fontDialog * f )
{
LRESULT pos ;
BOOL selected ;
selected = cbGetCurSel ( f - > sizeCombobox , & pos ) ;
if ( ! selected ) // on deselect, do nothing
return ;
f - > curSize = defaultSizes [ pos ] . value ;
queueRedrawSampleText ( f ) ;
}
static void sizeEdited ( struct fontDialog * f )
{
WCHAR * wsize ;
double size ;
// handle type-to-selection
if ( cbTypeToSelect ( f - > sizeCombobox , NULL , FALSE ) ) {
sizeChanged ( f ) ;
return ;
}
// selection not chosen, try to parse the typing
wsize = windowText ( f - > sizeCombobox ) ;
// this is what the Choose Font dialog does; it swallows errors while the real ChooseFont() is not lenient (and only checks on OK)
size = wcstod ( wsize , NULL ) ;
2017-02-11 15:55:30 -06:00
// TODO free wsize? I forget already
2016-04-17 19:10:14 -05:00
if ( size < = 0 ) // don't change on invalid size
return ;
f - > curSize = size ;
queueRedrawSampleText ( f ) ;
}
2016-04-17 12:06:45 -05:00
static void fontDialogDrawSampleText ( struct fontDialog * f , ID2D1RenderTarget * rt )
{
D2D1_COLOR_F color ;
D2D1_BRUSH_PROPERTIES props ;
ID2D1SolidColorBrush * black ;
IDWriteFont * font ;
IDWriteLocalizedStrings * sampleStrings ;
BOOL exists ;
WCHAR * sample ;
2016-04-17 19:10:14 -05:00
WCHAR * family ;
2016-04-17 12:06:45 -05:00
IDWriteTextFormat * format ;
D2D1_RECT_F rect ;
HRESULT hr ;
color . r = 0.0 ;
color . g = 0.0 ;
color . b = 0.0 ;
color . a = 1.0 ;
ZeroMemory ( & props , sizeof ( D2D1_BRUSH_PROPERTIES ) ) ;
props . opacity = 1.0 ;
// identity matrix
props . transform . _11 = 1 ;
props . transform . _22 = 1 ;
hr = rt - > CreateSolidColorBrush (
& color ,
& props ,
& black ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error creating solid brush " , hr ) ;
2016-04-17 12:06:45 -05:00
font = ( IDWriteFont * ) cbGetItemData ( f - > styleCombobox , ( WPARAM ) f - > curStyle ) ;
hr = font - > GetInformationalStrings ( DWRITE_INFORMATIONAL_STRING_SAMPLE_TEXT , & sampleStrings , & exists ) ;
if ( hr ! = S_OK )
exists = FALSE ;
if ( exists ) {
2018-03-17 14:49:00 -05:00
sample = uiprivFontCollectionCorrectString ( f - > fc , sampleStrings ) ;
2016-04-17 12:06:45 -05:00
sampleStrings - > Release ( ) ;
} else
2019-04-03 09:11:57 -05:00
sample = ( WCHAR * ) L " The quick brown fox jumps over the lazy dog. " ;
2016-04-17 12:06:45 -05:00
2016-04-17 19:10:14 -05:00
// DirectWrite doesn't allow creating a text format from a font; we need to get this ourselves
family = cbGetItemText ( f - > familyCombobox , f - > curFamily ) ;
2016-04-17 12:06:45 -05:00
hr = dwfactory - > CreateTextFormat ( family ,
NULL ,
font - > GetWeight ( ) ,
font - > GetStyle ( ) ,
font - > GetStretch ( ) ,
// typographic points are 1/72 inch; this parameter is 1/96 inch
// fortunately Microsoft does this too, in https://msdn.microsoft.com/en-us/library/windows/desktop/dd371554%28v=vs.85%29.aspx
2016-04-17 19:10:14 -05:00
f - > curSize * ( 96.0 / 72.0 ) ,
2016-04-17 12:06:45 -05:00
// see http://stackoverflow.com/questions/28397971/idwritefactorycreatetextformat-failing and https://msdn.microsoft.com/en-us/library/windows/desktop/dd368203.aspx
// TODO use the current locale again?
L " " ,
& format ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error creating IDWriteTextFormat " , hr ) ;
2018-04-15 17:12:58 -05:00
uiprivFree ( family ) ;
2016-04-17 12:06:45 -05:00
rect . left = 0 ;
rect . top = 0 ;
2016-06-05 15:07:40 -05:00
rect . right = realGetSize ( rt ) . width ;
rect . bottom = realGetSize ( rt ) . height ;
2016-04-17 12:06:45 -05:00
rt - > DrawText ( sample , wcslen ( sample ) ,
format ,
& rect ,
black ,
// TODO really?
D2D1_DRAW_TEXT_OPTIONS_NONE ,
DWRITE_MEASURING_MODE_NATURAL ) ;
format - > Release ( ) ;
if ( exists )
2018-04-15 17:12:58 -05:00
uiprivFree ( sample ) ;
2016-04-17 12:06:45 -05:00
black - > Release ( ) ;
}
static LRESULT CALLBACK fontDialogSampleSubProc ( HWND hwnd , UINT uMsg , WPARAM wParam , LPARAM lParam , UINT_PTR uIdSubclass , DWORD_PTR dwRefData )
{
ID2D1RenderTarget * rt ;
struct fontDialog * f ;
switch ( uMsg ) {
case msgD2DScratchPaint :
rt = ( ID2D1RenderTarget * ) lParam ;
f = ( struct fontDialog * ) dwRefData ;
fontDialogDrawSampleText ( f , rt ) ;
return 0 ;
case WM_NCDESTROY :
if ( RemoveWindowSubclass ( hwnd , fontDialogSampleSubProc , uIdSubclass ) = = FALSE )
2016-04-23 21:15:33 -05:00
logLastError ( L " error removing font dialog sample text subclass " ) ;
2016-04-17 12:06:45 -05:00
break ;
}
return DefSubclassProc ( hwnd , uMsg , wParam , lParam ) ;
2016-04-15 12:18:06 -05:00
}
2016-04-17 23:56:13 -05:00
static void setupInitialFontDialogState ( struct fontDialog * f )
{
WCHAR wsize [ 512 ] ; // this should be way more than enough
LRESULT pos ;
// first let's load the size
// the real font dialog:
// - if the chosen font size is in the list, it selects that item AND makes it topmost
// - if the chosen font size is not in the list, don't bother
// we'll simulate it by setting the text to a %f representation, then pretending as if it was entered
2016-04-18 16:14:33 -05:00
// TODO is 512 the correct number to pass to _snwprintf()?
// TODO will this revert to scientific notation?
_snwprintf ( wsize , 512 , L " %g " , f - > params - > size ) ;
2016-04-17 23:56:13 -05:00
// TODO make this a setWindowText()
if ( SendMessageW ( f - > sizeCombobox , WM_SETTEXT , 0 , ( LPARAM ) wsize ) ! = ( LRESULT ) TRUE )
2016-04-23 21:15:33 -05:00
logLastError ( L " error setting size combobox to initial font size " ) ;
2016-04-17 23:56:13 -05:00
sizeEdited ( f ) ;
if ( cbGetCurSel ( f - > sizeCombobox , & pos ) )
if ( SendMessageW ( f - > sizeCombobox , CB_SETTOPINDEX , ( WPARAM ) pos , 0 ) ! = 0 )
2016-04-23 21:15:33 -05:00
logLastError ( L " error making chosen size topmost in the size combobox " ) ;
2016-04-17 23:56:13 -05:00
// now we set the family and style
// we do this by first setting the previous style attributes, then simulating a font entered
2016-04-18 16:14:33 -05:00
f - > weight = f - > params - > font - > GetWeight ( ) ;
f - > style = f - > params - > font - > GetStyle ( ) ;
f - > stretch = f - > params - > font - > GetStretch ( ) ;
if ( SendMessageW ( f - > familyCombobox , WM_SETTEXT , 0 , ( LPARAM ) ( f - > params - > familyName ) ) ! = ( LRESULT ) TRUE )
2016-04-23 21:15:33 -05:00
logLastError ( L " error setting family combobox to initial font family " ) ;
2016-04-17 23:56:13 -05:00
familyEdited ( f ) ;
}
2016-04-14 20:35:50 -05:00
static struct fontDialog * beginFontDialog ( HWND hwnd , LPARAM lParam )
{
struct fontDialog * f ;
2016-04-17 00:34:05 -05:00
UINT32 i , nFamilies ;
IDWriteFontFamily * family ;
2016-04-14 21:25:32 -05:00
WCHAR * wname ;
2016-04-17 19:10:14 -05:00
LRESULT pos ;
2016-04-16 16:58:45 -05:00
HWND samplePlacement ;
2016-04-14 21:25:32 -05:00
HRESULT hr ;
2016-04-14 20:35:50 -05:00
2018-04-15 17:12:58 -05:00
f = uiprivNew ( struct fontDialog ) ;
2016-04-14 20:35:50 -05:00
f - > hwnd = hwnd ;
2016-04-18 12:32:48 -05:00
f - > params = ( struct fontDialogParams * ) lParam ;
2016-04-14 20:35:50 -05:00
2016-05-16 18:28:30 -05:00
f - > familyCombobox = getDlgItem ( f - > hwnd , rcFontFamilyCombobox ) ;
f - > styleCombobox = getDlgItem ( f - > hwnd , rcFontStyleCombobox ) ;
f - > sizeCombobox = getDlgItem ( f - > hwnd , rcFontSizeCombobox ) ;
2016-04-14 20:35:50 -05:00
2018-03-17 14:49:00 -05:00
f - > fc = uiprivLoadFontCollection ( ) ;
2016-04-17 00:34:05 -05:00
nFamilies = f - > fc - > fonts - > GetFontFamilyCount ( ) ;
for ( i = 0 ; i < nFamilies ; i + + ) {
hr = f - > fc - > fonts - > GetFontFamily ( i , & family ) ;
2016-04-14 21:25:32 -05:00
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error getting font family " , hr ) ;
2018-03-17 14:49:00 -05:00
wname = uiprivFontCollectionFamilyName ( f - > fc , family ) ;
2016-04-17 00:34:05 -05:00
pos = cbAddString ( f - > familyCombobox , wname ) ;
2018-04-15 17:12:58 -05:00
uiprivFree ( wname ) ;
2016-04-17 00:34:05 -05:00
cbSetItemData ( f - > familyCombobox , ( WPARAM ) pos , ( LPARAM ) family ) ;
2016-04-14 21:25:32 -05:00
}
2016-04-17 19:10:14 -05:00
for ( i = 0 ; defaultSizes [ i ] . text ! = NULL ; i + + )
cbInsertString ( f - > sizeCombobox , defaultSizes [ i ] . text , ( WPARAM ) i ) ;
2016-04-16 16:58:45 -05:00
2016-05-16 18:28:30 -05:00
samplePlacement = getDlgItem ( f - > hwnd , rcFontSamplePlacement ) ;
2016-04-29 16:08:31 -05:00
uiWindowsEnsureGetWindowRect ( samplePlacement , & ( f - > sampleRect ) ) ;
2016-04-16 16:58:45 -05:00
mapWindowRect ( NULL , f - > hwnd , & ( f - > sampleRect ) ) ;
2016-04-23 21:15:33 -05:00
uiWindowsEnsureDestroyWindow ( samplePlacement ) ;
2016-04-17 12:06:45 -05:00
f - > sampleBox = newD2DScratch ( f - > hwnd , & ( f - > sampleRect ) , ( HMENU ) rcFontSamplePlacement , fontDialogSampleSubProc , ( DWORD_PTR ) f ) ;
2016-04-16 16:58:45 -05:00
2016-04-17 23:56:13 -05:00
setupInitialFontDialogState ( f ) ;
2016-04-14 20:35:50 -05:00
return f ;
}
static void endFontDialog ( struct fontDialog * f , INT_PTR code )
{
2016-04-15 12:18:06 -05:00
wipeStylesBox ( f ) ;
2016-04-17 00:34:05 -05:00
cbWipeAndReleaseData ( f - > familyCombobox ) ;
2018-03-17 14:49:00 -05:00
uiprivFontCollectionFree ( f - > fc ) ;
2016-04-14 20:35:50 -05:00
if ( EndDialog ( f - > hwnd , code ) = = 0 )
2016-04-23 21:15:33 -05:00
logLastError ( L " error ending font dialog " ) ;
2018-04-15 17:12:58 -05:00
uiprivFree ( f ) ;
2016-04-14 20:35:50 -05:00
}
static INT_PTR tryFinishDialog ( struct fontDialog * f , WPARAM wParam )
{
2016-04-18 16:14:33 -05:00
IDWriteFontFamily * family ;
2016-04-18 00:12:25 -05:00
2016-04-14 20:35:50 -05:00
// cancelling
if ( LOWORD ( wParam ) ! = IDOK ) {
endFontDialog ( f , 1 ) ;
return TRUE ;
}
2016-04-17 21:09:18 -05:00
// OK
2018-03-17 14:49:00 -05:00
uiprivDestroyFontDialogParams ( f - > params ) ;
2016-04-18 16:14:33 -05:00
f - > params - > font = ( IDWriteFont * ) cbGetItemData ( f - > styleCombobox , f - > curStyle ) ;
// we need to save font from being destroyed with the combobox
f - > params - > font - > AddRef ( ) ;
f - > params - > size = f - > curSize ;
family = ( IDWriteFontFamily * ) cbGetItemData ( f - > familyCombobox , f - > curFamily ) ;
2018-03-17 14:49:00 -05:00
f - > params - > familyName = uiprivFontCollectionFamilyName ( f - > fc , family ) ;
2016-04-18 16:14:33 -05:00
f - > params - > styleName = fontStyleName ( f - > fc , f - > params - > font ) ;
2016-04-14 20:35:50 -05:00
endFontDialog ( f , 2 ) ;
return TRUE ;
}
static INT_PTR CALLBACK fontDialogDlgProc ( HWND hwnd , UINT uMsg , WPARAM wParam , LPARAM lParam )
{
struct fontDialog * f ;
f = ( struct fontDialog * ) GetWindowLongPtrW ( hwnd , DWLP_USER ) ;
if ( f = = NULL ) {
if ( uMsg = = WM_INITDIALOG ) {
f = beginFontDialog ( hwnd , lParam ) ;
SetWindowLongPtrW ( hwnd , DWLP_USER , ( LONG_PTR ) f ) ;
return TRUE ;
}
return FALSE ;
}
switch ( uMsg ) {
case WM_COMMAND :
2016-04-16 16:58:45 -05:00
SetWindowLongPtrW ( f - > hwnd , DWLP_MSGRESULT , 0 ) ; // just in case
2016-04-15 12:18:06 -05:00
switch ( LOWORD ( wParam ) ) {
case IDOK :
case IDCANCEL :
if ( HIWORD ( wParam ) ! = BN_CLICKED )
return FALSE ;
return tryFinishDialog ( f , wParam ) ;
case rcFontFamilyCombobox :
2016-04-17 18:03:34 -05:00
if ( HIWORD ( wParam ) = = CBN_SELCHANGE ) {
familyChanged ( f ) ;
return TRUE ;
}
if ( HIWORD ( wParam ) = = CBN_EDITCHANGE ) {
familyEdited ( f ) ;
return TRUE ;
}
return FALSE ;
2016-04-16 21:15:19 -05:00
case rcFontStyleCombobox :
2016-04-17 18:22:37 -05:00
if ( HIWORD ( wParam ) = = CBN_SELCHANGE ) {
styleChanged ( f ) ;
return TRUE ;
}
if ( HIWORD ( wParam ) = = CBN_EDITCHANGE ) {
styleEdited ( f ) ;
return TRUE ;
}
return FALSE ;
2016-04-16 21:15:19 -05:00
case rcFontSizeCombobox :
2016-04-17 19:10:14 -05:00
if ( HIWORD ( wParam ) = = CBN_SELCHANGE ) {
sizeChanged ( f ) ;
return TRUE ;
}
if ( HIWORD ( wParam ) = = CBN_EDITCHANGE ) {
sizeEdited ( f ) ;
return TRUE ;
}
return FALSE ;
2016-04-15 12:18:06 -05:00
}
return FALSE ;
2016-04-14 20:35:50 -05:00
}
return FALSE ;
}
2018-05-02 19:57:50 -05:00
// because Windows doesn't really support resources in static libraries, we have to embed this directly; oh well
/*
// this is for our custom DirectWrite-based font dialog (see fontdialog.cpp)
// this is based on the "New Font Dialog with Syslink" in Microsoft's font.dlg
// LONGTERM look at localization
// LONGTERM make it look tighter and nicer like the real one, including the actual heights of the font family and style comboboxes
rcFontDialog DIALOGEX 13 , 54 , 243 , 200
STYLE DS_MODALFRAME | WS_POPUP | WS_CAPTION | WS_SYSMENU | DS_3DLOOK
CAPTION " Font "
FONT 9 , " Segoe UI "
BEGIN
LTEXT " &Font: " , - 1 , 7 , 7 , 98 , 9
COMBOBOX rcFontFamilyCombobox , 7 , 16 , 98 , 76 ,
CBS_SIMPLE | CBS_AUTOHSCROLL | CBS_DISABLENOSCROLL |
CBS_SORT | WS_VSCROLL | WS_TABSTOP | CBS_HASSTRINGS
LTEXT " Font st&yle: " , - 1 , 114 , 7 , 74 , 9
COMBOBOX rcFontStyleCombobox , 114 , 16 , 74 , 76 ,
CBS_SIMPLE | CBS_AUTOHSCROLL | CBS_DISABLENOSCROLL |
WS_VSCROLL | WS_TABSTOP | CBS_HASSTRINGS
LTEXT " &Size: " , - 1 , 198 , 7 , 36 , 9
COMBOBOX rcFontSizeCombobox , 198 , 16 , 36 , 76 ,
CBS_SIMPLE | CBS_AUTOHSCROLL | CBS_DISABLENOSCROLL |
CBS_SORT | WS_VSCROLL | WS_TABSTOP | CBS_HASSTRINGS
GROUPBOX " Sample " , - 1 , 7 , 97 , 227 , 70 , WS_GROUP
CTEXT " AaBbYyZz " , rcFontSamplePlacement , 9 , 106 , 224 , 60 , SS_NOPREFIX | NOT WS_VISIBLE
DEFPUSHBUTTON " OK " , IDOK , 141 , 181 , 45 , 14 , WS_GROUP
PUSHBUTTON " Cancel " , IDCANCEL , 190 , 181 , 45 , 14 , WS_GROUP
END
*/
static const uint8_t data_rcFontDialog [ ] = {
0x01 , 0x00 , 0xFF , 0xFF , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0xC4 , 0x00 , 0xC8 , 0x80 ,
0x0A , 0x00 , 0x0D , 0x00 , 0x36 , 0x00 , 0xF3 , 0x00 ,
0xC8 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x46 , 0x00 ,
0x6F , 0x00 , 0x6E , 0x00 , 0x74 , 0x00 , 0x00 , 0x00 ,
0x09 , 0x00 , 0x00 , 0x00 , 0x00 , 0x01 , 0x53 , 0x00 ,
0x65 , 0x00 , 0x67 , 0x00 , 0x6F , 0x00 , 0x65 , 0x00 ,
0x20 , 0x00 , 0x55 , 0x00 , 0x49 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x02 , 0x50 , 0x07 , 0x00 , 0x07 , 0x00 ,
0x62 , 0x00 , 0x09 , 0x00 , 0xFF , 0xFF , 0xFF , 0xFF ,
0xFF , 0xFF , 0x82 , 0x00 , 0x26 , 0x00 , 0x46 , 0x00 ,
0x6F , 0x00 , 0x6E , 0x00 , 0x74 , 0x00 , 0x3A , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x41 , 0x0B , 0x21 , 0x50 ,
0x07 , 0x00 , 0x10 , 0x00 , 0x62 , 0x00 , 0x4C , 0x00 ,
0xE8 , 0x03 , 0x00 , 0x00 , 0xFF , 0xFF , 0x85 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x02 , 0x50 ,
0x72 , 0x00 , 0x07 , 0x00 , 0x4A , 0x00 , 0x09 , 0x00 ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0x82 , 0x00 ,
0x46 , 0x00 , 0x6F , 0x00 , 0x6E , 0x00 , 0x74 , 0x00 ,
0x20 , 0x00 , 0x73 , 0x00 , 0x74 , 0x00 , 0x26 , 0x00 ,
0x79 , 0x00 , 0x6C , 0x00 , 0x65 , 0x00 , 0x3A , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x41 , 0x0A , 0x21 , 0x50 ,
0x72 , 0x00 , 0x10 , 0x00 , 0x4A , 0x00 , 0x4C , 0x00 ,
0xE9 , 0x03 , 0x00 , 0x00 , 0xFF , 0xFF , 0x85 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x02 , 0x50 ,
0xC6 , 0x00 , 0x07 , 0x00 , 0x24 , 0x00 , 0x09 , 0x00 ,
0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0xFF , 0x82 , 0x00 ,
0x26 , 0x00 , 0x53 , 0x00 , 0x69 , 0x00 , 0x7A , 0x00 ,
0x65 , 0x00 , 0x3A , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x41 , 0x0B , 0x21 , 0x50 , 0xC6 , 0x00 , 0x10 , 0x00 ,
0x24 , 0x00 , 0x4C , 0x00 , 0xEA , 0x03 , 0x00 , 0x00 ,
0xFF , 0xFF , 0x85 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x07 , 0x00 , 0x02 , 0x50 , 0x07 , 0x00 , 0x61 , 0x00 ,
0xE3 , 0x00 , 0x46 , 0x00 , 0xFF , 0xFF , 0xFF , 0xFF ,
0xFF , 0xFF , 0x80 , 0x00 , 0x53 , 0x00 , 0x61 , 0x00 ,
0x6D , 0x00 , 0x70 , 0x00 , 0x6C , 0x00 , 0x65 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x81 , 0x00 , 0x02 , 0x40 ,
0x09 , 0x00 , 0x6A , 0x00 , 0xE0 , 0x00 , 0x3C , 0x00 ,
0xEB , 0x03 , 0x00 , 0x00 , 0xFF , 0xFF , 0x82 , 0x00 ,
0x41 , 0x00 , 0x61 , 0x00 , 0x42 , 0x00 , 0x62 , 0x00 ,
0x59 , 0x00 , 0x79 , 0x00 , 0x5A , 0x00 , 0x7A , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x01 , 0x00 , 0x03 , 0x50 ,
0x8D , 0x00 , 0xB5 , 0x00 , 0x2D , 0x00 , 0x0E , 0x00 ,
0x01 , 0x00 , 0x00 , 0x00 , 0xFF , 0xFF , 0x80 , 0x00 ,
0x4F , 0x00 , 0x4B , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x03 , 0x50 , 0xBE , 0x00 , 0xB5 , 0x00 ,
0x2D , 0x00 , 0x0E , 0x00 , 0x02 , 0x00 , 0x00 , 0x00 ,
0xFF , 0xFF , 0x80 , 0x00 , 0x43 , 0x00 , 0x61 , 0x00 ,
0x6E , 0x00 , 0x63 , 0x00 , 0x65 , 0x00 , 0x6C , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 ,
} ;
static_assert ( ARRAYSIZE ( data_rcFontDialog ) = = 476 , " wrong size for resource rcFontDialog " ) ;
2018-03-17 13:47:20 -05:00
BOOL uiprivShowFontDialog ( HWND parent , struct fontDialogParams * params )
2016-04-14 20:35:50 -05:00
{
2018-05-02 19:57:50 -05:00
switch ( DialogBoxIndirectParamW ( hInstance , ( const DLGTEMPLATE * ) data_rcFontDialog , parent , fontDialogDlgProc , ( LPARAM ) params ) ) {
2016-04-17 21:09:18 -05:00
case 1 : // cancel
return FALSE ;
case 2 : // ok
// make the compiler happy by putting the return after the switch
2016-04-14 20:35:50 -05:00
break ;
default :
2016-04-23 21:15:33 -05:00
logLastError ( L " error running font dialog " ) ;
2016-04-14 20:35:50 -05:00
}
2016-04-17 21:09:18 -05:00
return TRUE ;
2016-04-14 20:35:50 -05:00
}
2016-04-18 12:32:48 -05:00
static IDWriteFontFamily * tryFindFamily ( IDWriteFontCollection * fc , const WCHAR * name )
{
UINT32 index ;
BOOL exists ;
IDWriteFontFamily * family ;
HRESULT hr ;
hr = fc - > FindFamilyName ( name , & index , & exists ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error finding font family for font dialog " , hr ) ;
2016-04-18 12:32:48 -05:00
if ( ! exists )
return NULL ;
hr = fc - > GetFontFamily ( index , & family ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error extracting found font family for font dialog " , hr ) ;
2016-04-18 12:32:48 -05:00
return family ;
}
2018-03-17 13:47:20 -05:00
void uiprivLoadInitialFontDialogParams ( struct fontDialogParams * params )
2016-04-18 12:32:48 -05:00
{
struct fontCollection * fc ;
IDWriteFontFamily * family ;
IDWriteFont * font ;
HRESULT hr ;
// Our preferred font is Arial 10 Regular.
// 10 comes from the official font dialog.
// Arial Regular is a reasonable, if arbitrary, default; it's similar to the defaults on other systems.
// If Arial isn't found, we'll use Helvetica and then MS Sans Serif as fallbacks, and if not, we'll just grab the first font family in the collection.
// We need the correct localized name for Regular (and possibly Arial too? let's say yes to be safe), so let's grab the strings from DirectWrite instead of hardcoding them.
2018-03-17 14:49:00 -05:00
fc = uiprivLoadFontCollection ( ) ;
2016-04-18 12:32:48 -05:00
family = tryFindFamily ( fc - > fonts , L " Arial " ) ;
if ( family = = NULL ) {
family = tryFindFamily ( fc - > fonts , L " Helvetica " ) ;
if ( family = = NULL ) {
family = tryFindFamily ( fc - > fonts , L " MS Sans Serif " ) ;
if ( family = = NULL ) {
hr = fc - > fonts - > GetFontFamily ( 0 , & family ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error getting first font out of font collection (worst case scenario) " , hr ) ;
2016-04-18 12:32:48 -05:00
}
}
}
// next part is simple: just get the closest match to regular
hr = family - > GetFirstMatchingFont (
DWRITE_FONT_WEIGHT_NORMAL ,
DWRITE_FONT_STRETCH_NORMAL ,
DWRITE_FONT_STYLE_NORMAL ,
& font ) ;
if ( hr ! = S_OK )
2016-04-23 21:15:33 -05:00
logHRESULT ( L " error getting Regular font from Arial " , hr ) ;
2016-04-18 12:32:48 -05:00
2016-04-18 16:14:33 -05:00
params - > font = font ;
params - > size = 10 ;
2018-03-17 14:49:00 -05:00
params - > familyName = uiprivFontCollectionFamilyName ( fc , family ) ;
2016-04-18 16:14:33 -05:00
params - > styleName = fontStyleName ( fc , font ) ;
// don't release font; we still need it
2016-04-18 12:32:48 -05:00
family - > Release ( ) ;
2018-03-17 14:49:00 -05:00
uiprivFontCollectionFree ( fc ) ;
2016-04-18 12:32:48 -05:00
}
2016-04-18 16:14:33 -05:00
2018-03-17 13:47:20 -05:00
void uiprivDestroyFontDialogParams ( struct fontDialogParams * params )
2016-04-18 16:14:33 -05:00
{
params - > font - > Release ( ) ;
2018-03-17 13:47:20 -05:00
uiprivFree ( params - > familyName ) ;
uiprivFree ( params - > styleName ) ;
2016-04-18 16:14:33 -05:00
}
2018-03-17 13:47:20 -05:00
WCHAR * uiprivFontDialogParamsToString ( struct fontDialogParams * params )
2016-04-18 16:14:33 -05:00
{
WCHAR * text ;
// TODO dynamically allocate
2018-04-15 17:12:58 -05:00
text = ( WCHAR * ) uiprivAlloc ( 512 * sizeof ( WCHAR ) , " WCHAR[] " ) ;
2016-04-18 16:14:33 -05:00
_snwprintf ( text , 512 , L " %s %s %g " ,
params - > familyName ,
params - > styleName ,
params - > size ) ;
return text ;
}