|  | /* | 
|  | * 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 struct SHADERMODE | 
|  | { | 
|  | UINT displayStrID; | 
|  | const char* settingStr; | 
|  | } const D3D_VS_Modes[] = { | 
|  | {IDS_SHADER_MODE_HARDWARE,  "hardware"}, | 
|  | {IDS_SHADER_MODE_NONE,      "none"}, | 
|  | {0, 0} | 
|  | }; | 
|  |  | 
|  |  | 
|  | int 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 { | 
|  | SetWindowText(GetDlgItem(dialog, IDC_DESKTOP_WIDTH), "800"); | 
|  | SetWindowText(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) | 
|  | { | 
|  | unsigned int it; | 
|  | char* buf; | 
|  |  | 
|  | convert_x11_desktop_key(); | 
|  | update_gui_for_desktop_mode(dialog); | 
|  |  | 
|  | updating_ui = TRUE; | 
|  |  | 
|  | SendDlgItemMessage(dialog, IDC_DESKTOP_WIDTH, EM_LIMITTEXT, RES_MAXLEN, 0); | 
|  | SendDlgItemMessage(dialog, IDC_DESKTOP_HEIGHT, EM_LIMITTEXT, RES_MAXLEN, 0); | 
|  |  | 
|  | buf = get_reg_key(config_key, keypath("X11 Driver"), "DXGrab", "N"); | 
|  | if (IS_OPTION_TRUE(*buf)) | 
|  | CheckDlgButton(dialog, IDC_DX_MOUSE_GRAB, BST_CHECKED); | 
|  | else | 
|  | CheckDlgButton(dialog, IDC_DX_MOUSE_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); | 
|  |  | 
|  |  | 
|  | SendDlgItemMessage(dialog, IDC_D3D_VSHADER_MODE, CB_RESETCONTENT, 0, 0); | 
|  | for (it = 0; 0 != D3D_VS_Modes[it].displayStrID; ++it) { | 
|  | SendDlgItemMessageW (dialog, IDC_D3D_VSHADER_MODE, CB_ADDSTRING, 0, | 
|  | (LPARAM)load_string (D3D_VS_Modes[it].displayStrID)); | 
|  | } | 
|  | buf = get_reg_key(config_key, keypath("Direct3D"), "VertexShaderMode", "hardware"); | 
|  | for (it = 0; NULL != D3D_VS_Modes[it].settingStr; ++it) { | 
|  | if (strcmp(buf, D3D_VS_Modes[it].settingStr) == 0) { | 
|  | SendDlgItemMessage(dialog, IDC_D3D_VSHADER_MODE, CB_SETCURSEL, it, 0); | 
|  | break ; | 
|  | } | 
|  | } | 
|  | if (NULL == D3D_VS_Modes[it].settingStr) { | 
|  | WINE_ERR("Invalid Direct3D VertexShader Mode read from registry (%s)\n", buf); | 
|  | } | 
|  | HeapFree(GetProcessHeap(), 0, buf); | 
|  |  | 
|  | buf = get_reg_key(config_key, keypath("Direct3D"), "PixelShaderMode", "enabled"); | 
|  | if (!strcmp(buf, "enabled")) | 
|  | CheckDlgButton(dialog, IDC_D3D_PSHADER_MODE, BST_CHECKED); | 
|  | else | 
|  | CheckDlgButton(dialog, IDC_D3D_PSHADER_MODE, 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}; | 
|  | 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); | 
|  | } | 
|  | if (!height || !height[0]) { | 
|  | HeapFree(GetProcessHeap(), 0, height); | 
|  | height = strdupW(def_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_dx_mouse_grab_clicked(HWND dialog) { | 
|  | if (IsDlgButtonChecked(dialog, IDC_DX_MOUSE_GRAB) == BST_CHECKED) | 
|  | set_reg_key(config_key, keypath("X11 Driver"), "DXGrab", "Y"); | 
|  | else | 
|  | set_reg_key(config_key, keypath("X11 Driver"), "DXGrab", "N"); | 
|  | } | 
|  |  | 
|  | static void on_d3d_vshader_mode_changed(HWND dialog) { | 
|  | int selected_mode = SendDlgItemMessage(dialog, IDC_D3D_VSHADER_MODE, CB_GETCURSEL, 0, 0); | 
|  | set_reg_key(config_key, keypath("Direct3D"), "VertexShaderMode", | 
|  | D3D_VS_Modes[selected_mode].settingStr); | 
|  | } | 
|  |  | 
|  | static void on_d3d_pshader_mode_clicked(HWND dialog) { | 
|  | if (IsDlgButtonChecked(dialog, IDC_D3D_PSHADER_MODE) == BST_CHECKED) | 
|  | set_reg_key(config_key, keypath("Direct3D"), "PixelShaderMode", "enabled"); | 
|  | else | 
|  | set_reg_key(config_key, keypath("Direct3D"), "PixelShaderMode", "disabled"); | 
|  | } | 
|  | 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) | 
|  | { | 
|  | DWORD dwLogpixels; | 
|  | char szLogpixels[MAXBUFLEN]; | 
|  |  | 
|  | updating_ui = TRUE; | 
|  |  | 
|  | dwLogpixels = read_logpixels_reg(); | 
|  | WINE_TRACE("%u\n", dwLogpixels); | 
|  |  | 
|  | sprintf(szLogpixels, "%u", dwLogpixels); | 
|  | SetDlgItemText(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) | 
|  | { | 
|  | 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) | 
|  | { | 
|  | char buf[16]; | 
|  |  | 
|  | dpi = fixed_dpi; | 
|  | sprintf(buf, "%u", dpi); | 
|  | SetDlgItemText(hDlg, IDC_RES_DPIEDIT, buf); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (dpi >= MINDPI && dpi <= MAXDPI) | 
|  | { | 
|  | SendDlgItemMessage(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) | 
|  | { | 
|  | LOGFONT lf; | 
|  | HFONT hfont; | 
|  |  | 
|  | hfont = (HFONT)SendDlgItemMessage(hDlg, IDC_RES_FONT_PREVIEW, WM_GETFONT, 0, 0); | 
|  |  | 
|  | GetObject(hfont, sizeof(lf), &lf); | 
|  |  | 
|  | if (lstrcmp(lf.lfFaceName, "Tahoma") != 0) | 
|  | lstrcpy(lf.lfFaceName, "Tahoma"); | 
|  | else | 
|  | DeleteObject(hfont); | 
|  |  | 
|  | lf.lfHeight = MulDiv(-10, dpi, 72); | 
|  | hfont = CreateFontIndirect(&lf); | 
|  | SendDlgItemMessage(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; | 
|  | SendMessage(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; | 
|  | SendMessage(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_DX_MOUSE_GRAB:  on_dx_mouse_grab_clicked(hDlg); break; | 
|  | case IDC_D3D_PSHADER_MODE: on_d3d_pshader_mode_clicked(hDlg); break; | 
|  | } | 
|  | break; | 
|  | } | 
|  | case CBN_SELCHANGE: { | 
|  | SendMessage(GetParent(hDlg), PSM_CHANGED, 0, 0); | 
|  | switch (LOWORD(wParam)) { | 
|  | case IDC_D3D_VSHADER_MODE: on_d3d_vshader_mode_changed(hDlg); break; | 
|  | } | 
|  | break; | 
|  | } | 
|  |  | 
|  | default: | 
|  | break; | 
|  | } | 
|  | break; | 
|  |  | 
|  |  | 
|  | case WM_NOTIFY: | 
|  | switch (((LPNMHDR)lParam)->code) { | 
|  | case PSN_KILLACTIVE: { | 
|  | SetWindowLongPtr(hDlg, DWLP_MSGRESULT, FALSE); | 
|  | break; | 
|  | } | 
|  | case PSN_APPLY: { | 
|  | apply(); | 
|  | SetWindowLongPtr(hDlg, DWLP_MSGRESULT, PSNRET_NOERROR); | 
|  | break; | 
|  | } | 
|  | case PSN_SETACTIVE: { | 
|  | init_dialog (hDlg); | 
|  | break; | 
|  | } | 
|  | } | 
|  | break; | 
|  |  | 
|  | case WM_HSCROLL: | 
|  | switch (wParam) { | 
|  | default: { | 
|  | char buf[MAXBUFLEN]; | 
|  | int i = SendMessageW(GetDlgItem(hDlg, IDC_RES_TRACKBAR), TBM_GETPOS, 0, 0); | 
|  | buf[0] = 0; | 
|  | sprintf(buf, "%d", i); | 
|  | SendMessage(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; | 
|  | } |