443 lines
13 KiB
C
443 lines
13 KiB
C
/*
|
|
* Graphics configuration code
|
|
*
|
|
* Copyright 2003 Mark Westcott
|
|
* Copyright 2003-2004 Mike Hearn
|
|
* Copyright 2005 Raphael Junqueira
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
*
|
|
*/
|
|
|
|
#define WIN32_LEAN_AND_MEAN
|
|
|
|
#include <stdarg.h>
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
|
|
#include <windows.h>
|
|
#include <wine/unicode.h>
|
|
#include <wine/debug.h>
|
|
|
|
#include "resource.h"
|
|
#include "winecfg.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(winecfg);
|
|
|
|
#define RES_MAXLEN 5 /* max number of digits in a screen dimension. 5 digits should be plenty */
|
|
#define MINDPI 96
|
|
#define MAXDPI 480
|
|
#define DEFDPI 96
|
|
|
|
#define IDT_DPIEDIT 0x1234
|
|
|
|
static const WCHAR logpixels_reg[] = {'S','y','s','t','e','m','\\','C','u','r','r','e','n','t','C','o','n','t','r','o','l','S','e','t','\\','H','a','r','d','w','a','r','e',' ','P','r','o','f','i','l','e','s','\\','C','u','r','r','e','n','t','\\','S','o','f','t','w','a','r','e','\\','F','o','n','t','s',0};
|
|
static const WCHAR logpixels[] = {'L','o','g','P','i','x','e','l','s',0};
|
|
|
|
static const WCHAR desktopW[] = {'D','e','s','k','t','o','p',0};
|
|
static const WCHAR defaultW[] = {'D','e','f','a','u','l','t',0};
|
|
static const WCHAR explorerW[] = {'E','x','p','l','o','r','e','r',0};
|
|
static const WCHAR explorer_desktopsW[] = {'E','x','p','l','o','r','e','r','\\',
|
|
'D','e','s','k','t','o','p','s',0};
|
|
static const WCHAR x11_driverW[] = {'X','1','1',' ','D','r','i','v','e','r',0};
|
|
static const WCHAR default_resW[] = {'8','0','0','x','6','0','0',0};
|
|
|
|
|
|
static BOOL updating_ui;
|
|
|
|
/* convert the x11 desktop key to the new explorer config */
|
|
static void convert_x11_desktop_key(void)
|
|
{
|
|
char *buf;
|
|
|
|
if (!(buf = get_reg_key(config_key, "X11 Driver", "Desktop", NULL))) return;
|
|
set_reg_key(config_key, "Explorer\\Desktops", "Default", buf);
|
|
set_reg_key(config_key, "Explorer", "Desktop", "Default");
|
|
set_reg_key(config_key, "X11 Driver", "Desktop", NULL);
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
}
|
|
|
|
static void update_gui_for_desktop_mode(HWND dialog)
|
|
{
|
|
WCHAR *buf, *bufindex;
|
|
const WCHAR *desktop_name = current_app ? current_app : defaultW;
|
|
|
|
WINE_TRACE("\n");
|
|
updating_ui = TRUE;
|
|
|
|
buf = get_reg_keyW(config_key, explorer_desktopsW, desktop_name, NULL);
|
|
if (buf && (bufindex = strchrW(buf, 'x')))
|
|
{
|
|
*bufindex = 0;
|
|
++bufindex;
|
|
SetWindowTextW(GetDlgItem(dialog, IDC_DESKTOP_WIDTH), buf);
|
|
SetWindowTextW(GetDlgItem(dialog, IDC_DESKTOP_HEIGHT), bufindex);
|
|
} else {
|
|
SetWindowTextA(GetDlgItem(dialog, IDC_DESKTOP_WIDTH), "800");
|
|
SetWindowTextA(GetDlgItem(dialog, IDC_DESKTOP_HEIGHT), "600");
|
|
}
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
|
|
/* do we have desktop mode enabled? */
|
|
if (reg_key_exists(config_key, keypath("Explorer"), "Desktop"))
|
|
{
|
|
CheckDlgButton(dialog, IDC_ENABLE_DESKTOP, BST_CHECKED);
|
|
enable(IDC_DESKTOP_WIDTH);
|
|
enable(IDC_DESKTOP_HEIGHT);
|
|
enable(IDC_DESKTOP_SIZE);
|
|
enable(IDC_DESKTOP_BY);
|
|
}
|
|
else
|
|
{
|
|
CheckDlgButton(dialog, IDC_ENABLE_DESKTOP, BST_UNCHECKED);
|
|
disable(IDC_DESKTOP_WIDTH);
|
|
disable(IDC_DESKTOP_HEIGHT);
|
|
disable(IDC_DESKTOP_SIZE);
|
|
disable(IDC_DESKTOP_BY);
|
|
}
|
|
|
|
updating_ui = FALSE;
|
|
}
|
|
|
|
static void init_dialog(HWND dialog)
|
|
{
|
|
char* buf;
|
|
|
|
convert_x11_desktop_key();
|
|
update_gui_for_desktop_mode(dialog);
|
|
|
|
updating_ui = TRUE;
|
|
|
|
SendDlgItemMessageW(dialog, IDC_DESKTOP_WIDTH, EM_LIMITTEXT, RES_MAXLEN, 0);
|
|
SendDlgItemMessageW(dialog, IDC_DESKTOP_HEIGHT, EM_LIMITTEXT, RES_MAXLEN, 0);
|
|
|
|
buf = get_reg_key(config_key, keypath("X11 Driver"), "GrabFullscreen", "N");
|
|
if (IS_OPTION_TRUE(*buf))
|
|
CheckDlgButton(dialog, IDC_FULLSCREEN_GRAB, BST_CHECKED);
|
|
else
|
|
CheckDlgButton(dialog, IDC_FULLSCREEN_GRAB, BST_UNCHECKED);
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
|
|
buf = get_reg_key(config_key, keypath("X11 Driver"), "Managed", "Y");
|
|
if (IS_OPTION_TRUE(*buf))
|
|
CheckDlgButton(dialog, IDC_ENABLE_MANAGED, BST_CHECKED);
|
|
else
|
|
CheckDlgButton(dialog, IDC_ENABLE_MANAGED, BST_UNCHECKED);
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
|
|
buf = get_reg_key(config_key, keypath("X11 Driver"), "Decorated", "Y");
|
|
if (IS_OPTION_TRUE(*buf))
|
|
CheckDlgButton(dialog, IDC_ENABLE_DECORATED, BST_CHECKED);
|
|
else
|
|
CheckDlgButton(dialog, IDC_ENABLE_DECORATED, BST_UNCHECKED);
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
|
|
updating_ui = FALSE;
|
|
}
|
|
|
|
static void set_from_desktop_edits(HWND dialog)
|
|
{
|
|
static const WCHAR x[] = {'x',0};
|
|
static const WCHAR def_width[] = {'8','0','0',0};
|
|
static const WCHAR def_height[] = {'6','0','0',0};
|
|
static const WCHAR min_width[] = {'6','4','0',0};
|
|
static const WCHAR min_height[] = {'4','8','0',0};
|
|
WCHAR *width, *height, *new;
|
|
const WCHAR *desktop_name = current_app ? current_app : defaultW;
|
|
|
|
if (updating_ui) return;
|
|
|
|
WINE_TRACE("\n");
|
|
|
|
width = get_textW(dialog, IDC_DESKTOP_WIDTH);
|
|
height = get_textW(dialog, IDC_DESKTOP_HEIGHT);
|
|
|
|
if (!width || !width[0]) {
|
|
HeapFree(GetProcessHeap(), 0, width);
|
|
width = strdupW(def_width);
|
|
}
|
|
else if (atoiW(width) < atoiW(min_width))
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, width);
|
|
width = strdupW(min_width);
|
|
}
|
|
if (!height || !height[0]) {
|
|
HeapFree(GetProcessHeap(), 0, height);
|
|
height = strdupW(def_height);
|
|
}
|
|
else if (atoiW(height) < atoiW(min_height))
|
|
{
|
|
HeapFree(GetProcessHeap(), 0, height);
|
|
height = strdupW(min_height);
|
|
}
|
|
|
|
new = HeapAlloc(GetProcessHeap(), 0, (strlenW(width) + strlenW(height) + 2) * sizeof(WCHAR));
|
|
strcpyW( new, width );
|
|
strcatW( new, x );
|
|
strcatW( new, height );
|
|
set_reg_keyW(config_key, explorer_desktopsW, desktop_name, new);
|
|
set_reg_keyW(config_key, keypathW(explorerW), desktopW, desktop_name);
|
|
|
|
HeapFree(GetProcessHeap(), 0, width);
|
|
HeapFree(GetProcessHeap(), 0, height);
|
|
HeapFree(GetProcessHeap(), 0, new);
|
|
}
|
|
|
|
static void on_enable_desktop_clicked(HWND dialog) {
|
|
WINE_TRACE("\n");
|
|
|
|
if (IsDlgButtonChecked(dialog, IDC_ENABLE_DESKTOP) == BST_CHECKED) {
|
|
set_from_desktop_edits(dialog);
|
|
} else {
|
|
set_reg_key(config_key, keypath("Explorer"), "Desktop", NULL);
|
|
}
|
|
|
|
update_gui_for_desktop_mode(dialog);
|
|
}
|
|
|
|
static void on_enable_managed_clicked(HWND dialog) {
|
|
WINE_TRACE("\n");
|
|
|
|
if (IsDlgButtonChecked(dialog, IDC_ENABLE_MANAGED) == BST_CHECKED) {
|
|
set_reg_key(config_key, keypath("X11 Driver"), "Managed", "Y");
|
|
} else {
|
|
set_reg_key(config_key, keypath("X11 Driver"), "Managed", "N");
|
|
}
|
|
}
|
|
|
|
static void on_enable_decorated_clicked(HWND dialog) {
|
|
WINE_TRACE("\n");
|
|
|
|
if (IsDlgButtonChecked(dialog, IDC_ENABLE_DECORATED) == BST_CHECKED) {
|
|
set_reg_key(config_key, keypath("X11 Driver"), "Decorated", "Y");
|
|
} else {
|
|
set_reg_key(config_key, keypath("X11 Driver"), "Decorated", "N");
|
|
}
|
|
}
|
|
|
|
static void on_fullscreen_grab_clicked(HWND dialog)
|
|
{
|
|
if (IsDlgButtonChecked(dialog, IDC_FULLSCREEN_GRAB) == BST_CHECKED)
|
|
set_reg_key(config_key, keypath("X11 Driver"), "GrabFullscreen", "Y");
|
|
else
|
|
set_reg_key(config_key, keypath("X11 Driver"), "GrabFullscreen", "N");
|
|
}
|
|
|
|
static INT read_logpixels_reg(void)
|
|
{
|
|
DWORD dwLogPixels;
|
|
WCHAR *buf = get_reg_keyW(HKEY_LOCAL_MACHINE, logpixels_reg, logpixels, NULL);
|
|
dwLogPixels = buf ? *buf : DEFDPI;
|
|
HeapFree(GetProcessHeap(), 0, buf);
|
|
return dwLogPixels;
|
|
}
|
|
|
|
static void init_dpi_editbox(HWND hDlg)
|
|
{
|
|
static const WCHAR fmtW[] = {'%','u',0};
|
|
DWORD dwLogpixels;
|
|
WCHAR szLogpixels[MAXBUFLEN];
|
|
|
|
updating_ui = TRUE;
|
|
|
|
dwLogpixels = read_logpixels_reg();
|
|
WINE_TRACE("%u\n", dwLogpixels);
|
|
|
|
sprintfW(szLogpixels, fmtW, dwLogpixels);
|
|
SetDlgItemTextW(hDlg, IDC_RES_DPIEDIT, szLogpixels);
|
|
|
|
updating_ui = FALSE;
|
|
}
|
|
|
|
static void init_trackbar(HWND hDlg)
|
|
{
|
|
HWND hTrackBar = GetDlgItem(hDlg, IDC_RES_TRACKBAR);
|
|
DWORD dwLogpixels;
|
|
|
|
updating_ui = TRUE;
|
|
|
|
dwLogpixels = read_logpixels_reg();
|
|
|
|
SendMessageW(hTrackBar, TBM_SETRANGE, TRUE, MAKELONG(MINDPI, MAXDPI));
|
|
SendMessageW(hTrackBar, TBM_SETPOS, TRUE, dwLogpixels);
|
|
|
|
updating_ui = FALSE;
|
|
}
|
|
|
|
static void update_dpi_trackbar_from_edit(HWND hDlg, BOOL fix)
|
|
{
|
|
static const WCHAR fmtW[] = {'%','u',0};
|
|
DWORD dpi;
|
|
|
|
updating_ui = TRUE;
|
|
|
|
dpi = GetDlgItemInt(hDlg, IDC_RES_DPIEDIT, NULL, FALSE);
|
|
|
|
if (fix)
|
|
{
|
|
DWORD fixed_dpi = dpi;
|
|
|
|
if (dpi < MINDPI) fixed_dpi = MINDPI;
|
|
if (dpi > MAXDPI) fixed_dpi = MAXDPI;
|
|
|
|
if (fixed_dpi != dpi)
|
|
{
|
|
WCHAR buf[16];
|
|
|
|
dpi = fixed_dpi;
|
|
sprintfW(buf, fmtW, dpi);
|
|
SetDlgItemTextW(hDlg, IDC_RES_DPIEDIT, buf);
|
|
}
|
|
}
|
|
|
|
if (dpi >= MINDPI && dpi <= MAXDPI)
|
|
{
|
|
SendDlgItemMessageW(hDlg, IDC_RES_TRACKBAR, TBM_SETPOS, TRUE, dpi);
|
|
set_reg_key_dwordW(HKEY_LOCAL_MACHINE, logpixels_reg, logpixels, dpi);
|
|
}
|
|
|
|
updating_ui = FALSE;
|
|
}
|
|
|
|
static void update_font_preview(HWND hDlg)
|
|
{
|
|
DWORD dpi;
|
|
|
|
updating_ui = TRUE;
|
|
|
|
dpi = GetDlgItemInt(hDlg, IDC_RES_DPIEDIT, NULL, FALSE);
|
|
|
|
if (dpi >= MINDPI && dpi <= MAXDPI)
|
|
{
|
|
static const WCHAR tahomaW[] = {'T','a','h','o','m','a',0};
|
|
LOGFONTW lf;
|
|
HFONT hfont;
|
|
|
|
hfont = (HFONT)SendDlgItemMessageW(hDlg, IDC_RES_FONT_PREVIEW, WM_GETFONT, 0, 0);
|
|
|
|
GetObjectW(hfont, sizeof(lf), &lf);
|
|
|
|
if (strcmpW(lf.lfFaceName, tahomaW) != 0)
|
|
strcpyW(lf.lfFaceName, tahomaW);
|
|
else
|
|
DeleteObject(hfont);
|
|
lf.lfHeight = MulDiv(-10, dpi, 72);
|
|
hfont = CreateFontIndirectW(&lf);
|
|
SendDlgItemMessageW(hDlg, IDC_RES_FONT_PREVIEW, WM_SETFONT, (WPARAM)hfont, 1);
|
|
}
|
|
|
|
updating_ui = FALSE;
|
|
}
|
|
|
|
INT_PTR CALLBACK
|
|
GraphDlgProc(HWND hDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
switch (uMsg) {
|
|
case WM_INITDIALOG:
|
|
init_dpi_editbox(hDlg);
|
|
init_trackbar(hDlg);
|
|
update_font_preview(hDlg);
|
|
break;
|
|
|
|
case WM_SHOWWINDOW:
|
|
set_window_title(hDlg);
|
|
break;
|
|
|
|
case WM_TIMER:
|
|
if (wParam == IDT_DPIEDIT)
|
|
{
|
|
KillTimer(hDlg, IDT_DPIEDIT);
|
|
update_dpi_trackbar_from_edit(hDlg, TRUE);
|
|
update_font_preview(hDlg);
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch(HIWORD(wParam)) {
|
|
case EN_CHANGE: {
|
|
if (updating_ui) break;
|
|
SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
|
|
if ( ((LOWORD(wParam) == IDC_DESKTOP_WIDTH) || (LOWORD(wParam) == IDC_DESKTOP_HEIGHT)) && !updating_ui )
|
|
set_from_desktop_edits(hDlg);
|
|
else if (LOWORD(wParam) == IDC_RES_DPIEDIT)
|
|
{
|
|
update_dpi_trackbar_from_edit(hDlg, FALSE);
|
|
update_font_preview(hDlg);
|
|
SetTimer(hDlg, IDT_DPIEDIT, 1500, NULL);
|
|
}
|
|
break;
|
|
}
|
|
case BN_CLICKED: {
|
|
if (updating_ui) break;
|
|
SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
|
|
switch(LOWORD(wParam)) {
|
|
case IDC_ENABLE_DESKTOP: on_enable_desktop_clicked(hDlg); break;
|
|
case IDC_ENABLE_MANAGED: on_enable_managed_clicked(hDlg); break;
|
|
case IDC_ENABLE_DECORATED: on_enable_decorated_clicked(hDlg); break;
|
|
case IDC_FULLSCREEN_GRAB: on_fullscreen_grab_clicked(hDlg); break;
|
|
}
|
|
break;
|
|
}
|
|
case CBN_SELCHANGE: {
|
|
SendMessageW(GetParent(hDlg), PSM_CHANGED, 0, 0);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
|
|
case WM_NOTIFY:
|
|
switch (((LPNMHDR)lParam)->code) {
|
|
case PSN_KILLACTIVE: {
|
|
SetWindowLongPtrW(hDlg, DWLP_MSGRESULT, FALSE);
|
|
break;
|
|
}
|
|
case PSN_APPLY: {
|
|
apply();
|
|
SetWindowLongPtrW(hDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
|
|
break;
|
|
}
|
|
case PSN_SETACTIVE: {
|
|
init_dialog (hDlg);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_HSCROLL:
|
|
switch (wParam) {
|
|
default: {
|
|
static const WCHAR fmtW[] = {'%','d',0};
|
|
WCHAR buf[MAXBUFLEN];
|
|
int i = SendMessageW(GetDlgItem(hDlg, IDC_RES_TRACKBAR), TBM_GETPOS, 0, 0);
|
|
buf[0] = 0;
|
|
sprintfW(buf, fmtW, i);
|
|
SendMessageW(GetDlgItem(hDlg, IDC_RES_DPIEDIT), WM_SETTEXT, 0, (LPARAM) buf);
|
|
update_font_preview(hDlg);
|
|
set_reg_key_dwordW(HKEY_LOCAL_MACHINE, logpixels_reg, logpixels, i);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return FALSE;
|
|
}
|