///////////////////////////////////////////////////////////////////////// // $Id$ ///////////////////////////////////////////////////////////////////////// // // Copyright (C) 2009 Volker Ruppert // // 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 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 #include "win32dialog.h" #if BX_USE_TEXTCONFIG && defined(WIN32) && (BX_WITH_WIN32 || BX_WITH_SDL) #include "bochs.h" #include "win32res.h" #define ID_LABEL 100 #define ID_PARAM 1000 #define ID_BROWSE 2000 #define ID_UPDOWN 3000 // dialog item list code typedef struct _dlg_list_t { bx_list_c *list; UINT dlg_list_id; UINT dlg_base_id; struct _dlg_list_t *next; } dlg_list_t; HFONT DlgFont; WNDPROC DefEditWndProc; UINT nextDlgID; dlg_list_t *dlg_lists = NULL; bx_bool registerDlgList(UINT lid, bx_list_c *list) { dlg_list_t *dlg_list; int items; dlg_list = (dlg_list_t *)malloc(sizeof(dlg_list_t)); if (dlg_list == NULL) { return 0; } dlg_list->list = list; dlg_list->dlg_list_id = lid; dlg_list->dlg_base_id = nextDlgID; items = list->get_size(); nextDlgID += items; dlg_list->next = NULL; if (dlg_lists == NULL) { dlg_lists = dlg_list; } else { dlg_list_t *temp = dlg_lists; while (temp->next) { if (temp->list == dlg_list->list) { free(dlg_list); return 0; } temp = temp->next; } temp->next = dlg_list; } return dlg_list->dlg_base_id; } UINT findDlgListBaseID(bx_list_c *list) { dlg_list_t *dlg_list; for (dlg_list = dlg_lists; dlg_list; dlg_list = dlg_list->next) { if (list == dlg_list->list) { return dlg_list->dlg_base_id; } } return 0; } bx_param_c *findParamFromDlgID(UINT cid) { dlg_list_t *dlg_list; int i; for (dlg_list = dlg_lists; dlg_list; dlg_list = dlg_list->next) { if ((cid >= dlg_list->dlg_base_id) && (cid < (dlg_list->dlg_base_id + dlg_list->list->get_size()))) { i = cid - dlg_list->dlg_base_id; return dlg_list->list->get(i); } if (cid == dlg_list->dlg_list_id) { return dlg_list->list; } } return NULL; } UINT findDlgIDFromParam(bx_param_c *param) { dlg_list_t *dlg_list; bx_list_c *list; UINT cid; int i; list = (bx_list_c*)param->get_parent(); for (dlg_list = dlg_lists; dlg_list; dlg_list = dlg_list->next) { if (list == dlg_list->list) { cid = dlg_list->dlg_base_id; for (i = 0; i < list->get_size(); i++) { if (param == list->get(i)) { return (cid + i); } } } } return 0; } void cleanupDlgLists() { dlg_list_t *d, *next; if (dlg_lists) { d = dlg_lists; while (d != NULL) { next = d->next; free(d); d = next; } dlg_lists = NULL; } } // tooltips code HWND hwndTT, tt_hwndDlg; HHOOK tt_hhk; const char *tt_text; BOOL CALLBACK EnumChildProc(HWND hwndCtrl, LPARAM lParam); LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam); BOOL CreateParamDlgTooltip(HWND hwndDlg) { InitCommonControls(); tt_hwndDlg = hwndDlg; hwndTT = CreateWindowEx(0, TOOLTIPS_CLASS, (LPSTR) NULL, TTS_ALWAYSTIP, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, tt_hwndDlg, (HMENU) NULL, NULL, NULL); if (hwndTT == NULL) return FALSE; if (!EnumChildWindows(tt_hwndDlg, (WNDENUMPROC) EnumChildProc, 0)) return FALSE; tt_hhk = SetWindowsHookEx(WH_GETMESSAGE, GetMsgProc, (HINSTANCE) NULL, GetCurrentThreadId()); if (tt_hhk == (HHOOK) NULL) return FALSE; return TRUE; } BOOL CALLBACK EnumChildProc(HWND hwndCtrl, LPARAM lParam) { TOOLINFO ti; char szClass[64]; GetClassName(hwndCtrl, szClass, sizeof(szClass)); if (lstrcmp(szClass, "STATIC")) { ti.cbSize = sizeof(TOOLINFO); ti.uFlags = TTF_IDISHWND; ti.hwnd = tt_hwndDlg; ti.uId = (UINT_PTR) hwndCtrl; ti.hinst = 0; ti.lpszText = LPSTR_TEXTCALLBACK; SendMessage(hwndTT, TTM_ADDTOOL, 0, (LPARAM) (LPTOOLINFO) &ti); } return TRUE; } LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam) { MSG *lpmsg; lpmsg = (MSG *) lParam; if (nCode < 0 || !(IsChild(tt_hwndDlg, lpmsg->hwnd))) return (CallNextHookEx(tt_hhk, nCode, wParam, lParam)); switch (lpmsg->message) { case WM_MOUSEMOVE: case WM_LBUTTONDOWN: case WM_LBUTTONUP: case WM_RBUTTONDOWN: case WM_RBUTTONUP: if (hwndTT != NULL) { MSG msg; msg.lParam = lpmsg->lParam; msg.wParam = lpmsg->wParam; msg.message = lpmsg->message; msg.hwnd = lpmsg->hwnd; SendMessage(hwndTT, TTM_RELAYEVENT, 0, (LPARAM) (LPMSG) &msg); } break; default: break; } return (CallNextHookEx(tt_hhk, nCode, wParam, lParam)); } VOID OnWMNotify(LPARAM lParam) { LPTOOLTIPTEXT lpttt; int idCtrl; if ((((LPNMHDR) lParam)->code) == TTN_NEEDTEXT) { idCtrl = GetDlgCtrlID((HWND) ((LPNMHDR) lParam)->idFrom); lpttt = (LPTOOLTIPTEXT) lParam; if ((idCtrl >= ID_PARAM) && (idCtrl < ID_BROWSE)) { bx_param_c *param = findParamFromDlgID(idCtrl - ID_PARAM); if (param != NULL) { tt_text = param->get_description(); if (lstrlen(tt_text) > 0) { lpttt->lpszText = (LPSTR)tt_text; } } } } } // gui functions int AskFilename(HWND hwnd, bx_param_filename_c *param, char *buffer) { OPENFILENAME ofn; int ret; DWORD errcode; char filename[BX_PATHNAME_LEN]; const char *title, *ext; char errtext[80]; if (buffer != NULL) { lstrcpyn(filename, buffer, BX_PATHNAME_LEN); } else { param->get(filename, BX_PATHNAME_LEN); } // common file dialogs don't accept raw device names if ((isalpha(filename[0])) && (filename[1] == ':') && (strlen(filename) == 2)) { filename[0] = 0; } title = param->get_label(); ext = param->get_extension(); if (!title) title = param->get_name(); memset(&ofn, 0, sizeof(OPENFILENAME)); ofn.lStructSize = sizeof(OPENFILENAME); ofn.hwndOwner = hwnd; ofn.lpstrFile = filename; ofn.nMaxFile = MAX_PATH; ofn.lpstrInitialDir = bx_startup_flags.initial_dir; ofn.lpstrTitle = title; ofn.Flags = OFN_EXPLORER | OFN_HIDEREADONLY; ofn.lpstrDefExt = ext; if (!lstrcmp(ext, "bxrc")) { ofn.lpstrFilter = "Bochs config files (*.bxrc)\0*.bxrc\0All files (*.*)\0*.*\0"; } else if (!lstrcmp(ext, "img")) { ofn.lpstrFilter = "Disk image files (*.img)\0*.img\0All files (*.*)\0*.*\0"; } else if (!lstrcmp(ext, "iso")) { ofn.lpstrFilter = "CD-ROM image files (*.iso)\0*.iso\0All files (*.*)\0*.*\0"; } else if (!lstrcmp(ext, "log")) { ofn.lpstrFilter = "Log files (*.log)\0*.log\0All files (*.*)\0*.*\0"; } else if (!lstrcmp(ext, "map")) { ofn.lpstrFilter = "Keymap files (*.map)\0*.map\0All files (*.*)\0*.*\0"; } else if (!lstrcmp(ext, "txt")) { ofn.lpstrFilter = "Text files (*.txt)\0*.txt\0All files (*.*)\0*.*\0"; } else if (!lstrcmp(ext, "bmp")) { ofn.lpstrFilter = "Windows bitmap files (*.bmp)\0*.bmp\0All files (*.*)\0*.*\0"; } else { ofn.lpstrFilter = "All files (*.*)\0*.*\0"; } if (param->get_options() & param->SAVE_FILE_DIALOG) { ofn.Flags |= OFN_OVERWRITEPROMPT; ret = GetSaveFileName(&ofn); } else { ofn.Flags |= OFN_FILEMUSTEXIST; ret = GetOpenFileName(&ofn); } if (buffer != NULL) { lstrcpyn(buffer, filename, BX_PATHNAME_LEN); } else { param->set(filename); } if (ret == 0) { errcode = CommDlgExtendedError(); if (errcode == 0) { ret = -1; } else { if (errcode == 0x3002) { wsprintf(errtext, "CommDlgExtendedError: invalid filename"); } else { wsprintf(errtext, "CommDlgExtendedError returns 0x%04x", errcode); } MessageBox(hwnd, errtext, "Error", MB_ICONERROR); } } return ret; } void InitDlgFont(void) { LOGFONT LFont; LFont.lfHeight = 8; // Default logical height of font LFont.lfWidth = 0; // Default logical average character width LFont.lfEscapement = 0; // angle of escapement LFont.lfOrientation = 0; // base-line orientation angle LFont.lfWeight = FW_NORMAL; // font weight LFont.lfItalic = 0; // italic attribute flag LFont.lfUnderline = 0; // underline attribute flag LFont.lfStrikeOut = 0; // strikeout attribute flag LFont.lfCharSet = DEFAULT_CHARSET; // character set identifier LFont.lfOutPrecision = OUT_DEFAULT_PRECIS; // output precision LFont.lfClipPrecision = CLIP_DEFAULT_PRECIS; // clipping precision LFont.lfQuality = DEFAULT_QUALITY; // output quality LFont.lfPitchAndFamily = DEFAULT_PITCH; // pitch and family lstrcpy(LFont.lfFaceName, "Helv"); // pointer to typeface name string DlgFont = CreateFontIndirect(&LFont); } LRESULT CALLBACK EditHexWndProc(HWND Window, UINT msg, WPARAM wParam, LPARAM lParam) { if (msg == WM_CHAR) { switch (wParam) { case 0x08: break; case 'x': break; default: if ((wParam < '0') || ((wParam > '9') && (wParam < 'A')) || ((wParam > 'F') && (wParam < 'a')) || (wParam > 'f')) { MessageBeep(MB_OK); return 0; } } } return CallWindowProc(DefEditWndProc, Window, msg, wParam, lParam); } HWND CreateLabel(HWND hDlg, UINT cid, UINT xpos, UINT ypos, UINT width, BOOL hide, const char *text) { HWND Label; RECT r; int code; code = ID_LABEL + cid; r.left = xpos; r.top = ypos + 2; r.right = r.left + width; r.bottom = r.top + 16; MapDialogRect(hDlg, &r); Label = CreateWindow("STATIC", text, WS_CHILD, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); SendMessage(Label, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(Label, hide ? SW_HIDE : SW_SHOW); return Label; } HWND CreateGroupbox(HWND hDlg, UINT cid, UINT xpos, UINT ypos, SIZE size, BOOL hide, bx_list_c *list) { HWND Groupbox; RECT r; int code; const char *title = NULL; code = ID_PARAM + cid; r.left = xpos; r.top = ypos; r.right = r.left + size.cx; r.bottom = r.top + size.cy; MapDialogRect(hDlg, &r); if (list->get_options() & list->USE_BOX_TITLE) { title = list->get_title(); } Groupbox = CreateWindow("BUTTON", title, BS_GROUPBOX | WS_CHILD, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); SendMessage(Groupbox, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(Groupbox, hide ? SW_HIDE : SW_SHOW); return Groupbox; } HWND CreateTabControl(HWND hDlg, UINT cid, UINT xpos, UINT ypos, SIZE size, BOOL hide, bx_list_c *list) { HWND TabControl; TC_ITEM tie; RECT r; int code, i; bx_param_c *item; code = ID_PARAM + cid; r.left = xpos; r.top = ypos; r.right = r.left + size.cx; r.bottom = r.top + size.cy; MapDialogRect(hDlg, &r); TabControl = CreateWindow(WC_TABCONTROL, "", WS_CHILD | WS_CLIPSIBLINGS | WS_VISIBLE, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); for (i = 0; i < list->get_size(); i++) { item = list->get(i); if (item->get_type() == BXT_LIST) { tie.mask = TCIF_TEXT; tie.pszText = ((bx_list_c*)item)->get_title(); TabCtrl_InsertItem(TabControl, i, &tie); } } TabCtrl_SetCurSel(TabControl, 0); SendMessage(TabControl, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(TabControl, hide ? SW_HIDE : SW_SHOW); return TabControl; } HWND CreateBrowseButton(HWND hDlg, UINT cid, UINT xpos, UINT ypos, BOOL hide) { HWND Button; RECT r; int code; code = ID_BROWSE + cid; r.left = xpos; r.top = ypos; r.right = r.left + 50; r.bottom = r.top + 14; MapDialogRect(hDlg, &r); Button = CreateWindow("BUTTON", "Browse...", WS_CHILD, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); SendMessage(Button, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(Button, hide ? SW_HIDE : SW_SHOW); return Button; } HWND CreateCheckbox(HWND hDlg, UINT cid, UINT xpos, UINT ypos, BOOL hide, bx_param_bool_c *bparam) { HWND Checkbox; RECT r; int code, val; code = ID_PARAM + cid; r.left = xpos; r.top = ypos; r.right = r.left + 20; r.bottom = r.top + 14; MapDialogRect(hDlg, &r); Checkbox = CreateWindow("BUTTON", "", BS_AUTOCHECKBOX | WS_CHILD | WS_TABSTOP, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); val = bparam->get(); SendMessage(Checkbox, BM_SETCHECK, val ? BST_CHECKED : BST_UNCHECKED, 0); SendMessage(Checkbox, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(Checkbox, hide ? SW_HIDE : SW_SHOW); return Checkbox; } HWND CreateInput(HWND hDlg, UINT cid, UINT xpos, UINT ypos, BOOL hide, bx_param_c *param) { HWND Input, Updown; RECT r; int code, i, style; bx_param_num_c *nparam = NULL; bx_param_string_c *sparam; char buffer[512]; char eachbyte[16]; char sep_string[2]; char *val; BOOL spinctrl = FALSE, hexedit = FALSE; code = ID_PARAM + cid; style = WS_CHILD | WS_TABSTOP; if (param->get_type() == BXT_PARAM_STRING) { sparam = (bx_param_string_c*)param; val = sparam->getptr(); if (sparam->get_options() & sparam->RAW_BYTES) { buffer[0] = 0; sep_string[0] = sparam->get_separator(); sep_string[1] = 0; for (i = 0; i < sparam->get_maxsize(); i++) { wsprintf(eachbyte, "%s%02x", (i>0)?sep_string : "", (Bit8u)0xff&val[i]); strncat(buffer, eachbyte, sizeof(buffer)); } } else { lstrcpyn(buffer, val, 512); style |= ES_AUTOHSCROLL; } } else { nparam = (bx_param_num_c*)param; if (nparam->get_base() == BASE_HEX) { wsprintf(buffer, "0x%x", nparam->get()); hexedit = TRUE; } else { wsprintf(buffer, "%d", nparam->get()); style |= ES_NUMBER; } if (nparam->get_options() & nparam->USE_SPIN_CONTROL) { spinctrl = TRUE; } } r.left = xpos; r.top = ypos; r.right = r.left + 100; r.bottom = r.top + 14; MapDialogRect(hDlg, &r); Input = CreateWindowEx(WS_EX_CLIENTEDGE | WS_EX_NOPARENTNOTIFY, "EDIT", buffer, style, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); if (hexedit) { DefEditWndProc = (WNDPROC)SetWindowLongPtr(Input, GWLP_WNDPROC, (LONG_PTR)EditHexWndProc); } if (spinctrl) { style = WS_CHILD | WS_BORDER | WS_VISIBLE | UDS_NOTHOUSANDS | UDS_ARROWKEYS | UDS_ALIGNRIGHT | UDS_SETBUDDYINT; Updown = CreateUpDownControl(style, 0, 0, 0, 0, hDlg, ID_UPDOWN + cid, NULL, Input, (int)nparam->get_max(), (int)nparam->get_min(), (int)nparam->get()); ShowWindow(Updown, hide ? SW_HIDE : SW_SHOW); } SendMessage(Input, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(Input, hide ? SW_HIDE : SW_SHOW); return Input; } HWND CreateCombobox(HWND hDlg, UINT cid, UINT xpos, UINT ypos, BOOL hide, bx_param_enum_c *eparam) { HWND Combo; RECT r; int code, j; const char *choice; code = ID_PARAM + cid; r.left = xpos; r.top = ypos; r.right = r.left + 100; r.bottom = r.top + 14 * ((int)(eparam->get_max() - eparam->get_min()) + 1); MapDialogRect(hDlg, &r); Combo = CreateWindow("COMBOBOX", "", WS_CHILD | WS_TABSTOP | WS_VSCROLL | CBS_DROPDOWNLIST, r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, hDlg, (HMENU)code, NULL, NULL); j = 0; while(1) { choice = eparam->get_choice(j); if (choice == NULL) break; SendMessage(Combo, CB_ADDSTRING, 0, (LPARAM)choice); j++; } SendMessage(Combo, CB_SETCURSEL, (WPARAM)(eparam->get()-eparam->get_min()), 0); SendMessage(Combo, WM_SETFONT, (WPARAM)DlgFont, TRUE); ShowWindow(Combo, hide ? SW_HIDE : SW_SHOW); return Combo; } void EnableParam(HWND hDlg, UINT cid, bx_param_c *param, BOOL val) { HWND Button, Updown; if (cid == 0) { cid = findDlgIDFromParam(param); } if (param->get_type() != BXT_LIST) { EnableWindow(GetDlgItem(hDlg, ID_LABEL + cid), val); EnableWindow(GetDlgItem(hDlg, ID_PARAM + cid), val); Button = GetDlgItem(hDlg, ID_BROWSE + cid); if (Button != NULL) { EnableWindow(Button, val); } Updown = GetDlgItem(hDlg, ID_UPDOWN + cid); if (Updown != NULL) { EnableWindow(Updown, val); } } } UINT GetLabelText(bx_param_c *param, bx_list_c *list, char *buffer) { const char *label, *tmpstr; char tmpbuf[512]; int len; label = param->get_label(); if (label == NULL) { label = param->get_name(); } if ((list->get_options() & list->SHOW_GROUP_NAME) && (param->get_group() != NULL)) { wsprintf(tmpbuf, "%s %s", param->get_group(), label); } else { lstrcpyn(tmpbuf, label, 512); } if (buffer != NULL) { lstrcpy(buffer, tmpbuf); } len = lstrlen(tmpbuf); tmpstr = strchr(tmpbuf, 10); if (tmpstr != NULL) { if (lstrlen(tmpstr) > (len / 2)) { len = lstrlen(tmpstr) - 1; } else { len -= lstrlen(tmpstr); } } return (len * 4); } SIZE CreateParamList(HWND hDlg, UINT lid, UINT xpos, UINT ypos, BOOL hide, bx_list_c *list) { SIZE size, lsize; bx_param_c *param; bx_param_string_c *sparam; char buffer[512]; int options; UINT cid, i, items, lw, w1, x0, x1, x2, y; BOOL ihide; items = list->get_size(); options = list->get_options(); cid = registerDlgList(lid, list); x0 = xpos + 5; size.cx = 195; if (options & list->USE_TAB_WINDOW) { y = ypos + 15; size.cy = 18; } else { y = ypos + 10; size.cy = 13; } // find out longest label text w1 = 78; for (i = 0; i < items; i++) { param = list->get(i); if (!SIM->get_init_done() || (param->get_enabled() && param->get_runtime_param())) { if (param->get_type() != BXT_LIST) { lw = GetLabelText(param, list, NULL); if (lw > w1) { w1 = lw; } } } } // set columns and width x1 = x0 + w1 + 2; x2 = x1 + 110; if (size.cx < (int)(x2 + 5)) { size.cx = x2 + 5; } // create controls for (i = 0; i < items; i++) { param = list->get(i); if (!SIM->get_init_done() || (param->get_enabled() && param->get_runtime_param())) { ihide = hide || ((i != 0) && (options & list->USE_TAB_WINDOW)); if (param->get_type() == BXT_LIST) { lsize = CreateParamList(hDlg, cid, x0 + 4, y + 1, ihide, (bx_list_c*)param); if ((lsize.cx + 18) > size.cx) { size.cx = lsize.cx + 18; } if (!(options & list->USE_TAB_WINDOW)) { y += (lsize.cy + 6); size.cy += (lsize.cy + 6); } else { if ((lsize.cy + 24) > size.cy) { size.cy = lsize.cy + 24; } } } else { lw = GetLabelText(param, list, buffer); /* HWND ltext = */ CreateLabel(hDlg, cid, x0, y, w1, hide, buffer); if (param->get_type() == BXT_PARAM_BOOL) { /* HWND control = */ CreateCheckbox(hDlg, cid, x1, y, hide, (bx_param_bool_c*)param); } else if (param->get_type() == BXT_PARAM_ENUM) { /* HWND control = */ CreateCombobox(hDlg, cid, x1, y, hide, (bx_param_enum_c*)param); } else if (param->get_type() == BXT_PARAM_NUM) { /* HWND control = */ CreateInput(hDlg, cid, x1, y, hide, param); } else if (param->get_type() == BXT_PARAM_STRING) { /* HWND control = */ CreateInput(hDlg, cid, x1, y, hide, param); sparam = (bx_param_string_c*)param; if (sparam->get_options() & sparam->IS_FILENAME) { /* HWND browse = */ CreateBrowseButton(hDlg, cid, x2, y, hide); if (size.cx < (int)(x2 + 60)) { size.cx = x2 + 60; } } } if (!param->get_enabled()) { EnableParam(hDlg, cid, param, FALSE); } y += 20; size.cy += 20; } } cid++; } if (options & list->USE_TAB_WINDOW) { CreateTabControl(hDlg, lid, xpos, ypos, size, hide, list); } else { CreateGroupbox(hDlg, lid, xpos, ypos, size, hide, list); } return size; } void SetParamList(HWND hDlg, bx_list_c *list) { bx_param_c *param; bx_param_num_c *nparam; bx_param_enum_c *eparam; bx_param_string_c *sparam; int j; LRESULT val; const char *src; char buffer[512], rawbuf[512]; UINT cid, i, items, lid; lid = findDlgListBaseID(list); items = list->get_size(); for (i = 0; i < items; i++) { cid = lid + i; param = list->get(i); if (param->get_enabled() && (!SIM->get_init_done() || (SIM->get_init_done() && param->get_runtime_param()))) { if (param->get_type() == BXT_LIST) { SetParamList(hDlg, (bx_list_c*)param); } else if (param->get_type() == BXT_PARAM_BOOL) { val = SendMessage(GetDlgItem(hDlg, ID_PARAM + cid), BM_GETCHECK, 0, 0); ((bx_param_bool_c*)param)->set(val == BST_CHECKED); } else if (param->get_type() == BXT_PARAM_ENUM) { val = SendMessage(GetDlgItem(hDlg, ID_PARAM + cid), CB_GETCURSEL, 0, 0); eparam = (bx_param_enum_c*)param; eparam->set(val + eparam->get_min()); } else { if (SendMessage(GetDlgItem(hDlg, ID_PARAM + cid), EM_GETMODIFY, 0, 0)) { if (param->get_type() == BXT_PARAM_NUM) { nparam = (bx_param_num_c*)param; if (nparam->get_base() == BASE_HEX) { GetWindowText(GetDlgItem(hDlg, ID_PARAM + cid), buffer, 511); sscanf(buffer, "%x", &val); } else { val = GetDlgItemInt(hDlg, ID_PARAM + cid, NULL, FALSE); } nparam->set(val); } else if (param->get_type() == BXT_PARAM_STRING) { GetWindowText(GetDlgItem(hDlg, ID_PARAM + cid), buffer, 511); sparam = (bx_param_string_c*)param; if (sparam->get_options() & sparam->RAW_BYTES) { src = &buffer[0]; memset(rawbuf, 0, sparam->get_maxsize()); for (j = 0; j < sparam->get_maxsize(); j++) { while (*src == sparam->get_separator()) src++; if (*src == 0) break; if (sscanf(src, "%02x", &val)) { rawbuf[j] = (char) val; src += 2; } else { break; } } sparam->set(rawbuf); } else { sparam->set(buffer); } } } } } if ((i + 1) >= (UINT)list->get_size()) break; } } void ShowParamList(HWND hDlg, UINT lid, BOOL show, bx_list_c *list) { UINT cid; int i; HWND Button, Updown; BOOL ishow; ShowWindow(GetDlgItem(hDlg, ID_PARAM + lid), show ? SW_SHOW : SW_HIDE); cid = findDlgListBaseID(list); for (i = 0; i < list->get_size(); i++) { if (list->get(i)->get_type() == BXT_LIST) { ishow = show; if (list->get_options() & list->USE_TAB_WINDOW) { ishow &= (i == 0); } ShowParamList(hDlg, cid + i, ishow, (bx_list_c*)list->get(i)); } else { ShowWindow(GetDlgItem(hDlg, ID_LABEL + cid + i), show ? SW_SHOW : SW_HIDE); ShowWindow(GetDlgItem(hDlg, ID_PARAM + cid + i), show ? SW_SHOW : SW_HIDE); Button = GetDlgItem(hDlg, ID_BROWSE + cid + i); if (Button != NULL) { ShowWindow(Button, show ? SW_SHOW : SW_HIDE); } Updown = GetDlgItem(hDlg, ID_UPDOWN + cid + i); if (Updown != NULL) { ShowWindow(Updown, show ? SW_SHOW : SW_HIDE); } } } } void ProcessDependentList(HWND hDlg, bx_param_c *param, BOOL enabled) { UINT cid; bx_list_c *deplist; bx_param_c *dparam; bx_param_enum_c *eparam; Bit64s value; Bit64u enable_bitmap, mask; char buffer[BX_PATHNAME_LEN]; int i; BOOL en; deplist = param->get_dependent_list(); if (deplist != NULL) { cid = findDlgIDFromParam(param); if (param->get_type() == BXT_PARAM_ENUM) { eparam = (bx_param_enum_c*)param; value = SendMessage(GetDlgItem(hDlg, ID_PARAM + cid), CB_GETCURSEL, 0, 0); enable_bitmap = eparam->get_dependent_bitmap(value + eparam->get_min()); mask = 0x1; for (i = 0; i < deplist->get_size(); i++) { dparam = deplist->get(i); if (dparam != param) { en = (enable_bitmap & mask) && enabled; cid = findDlgIDFromParam(dparam); if (en != IsWindowEnabled(GetDlgItem(hDlg, ID_PARAM + cid))) { ProcessDependentList(hDlg, dparam, en); EnableParam(hDlg, 0, dparam, en); } } mask <<= 1; } } else if ((param->get_type() == BXT_PARAM_BOOL) || (param->get_type() == BXT_PARAM_NUM) || (param->get_type() == BXT_PARAM_STRING)) { if (param->get_type() == BXT_PARAM_BOOL) { value = SendMessage(GetDlgItem(hDlg, ID_PARAM + cid), BM_GETCHECK, 0, 0); } else if (param->get_type() == BXT_PARAM_NUM) { value = GetDlgItemInt(hDlg, ID_PARAM + cid, NULL, FALSE); } else { GetWindowText(GetDlgItem(hDlg, ID_PARAM + cid), buffer, BX_PATHNAME_LEN); value = (lstrlen(buffer) > 0) && (strcmp(buffer, "none")); } for (i = 0; i < deplist->get_size(); i++) { dparam = deplist->get(i); if (dparam != param) { en = (value && enabled); cid = findDlgIDFromParam(dparam); if (en != IsWindowEnabled(GetDlgItem(hDlg, ID_PARAM + cid))) { ProcessDependentList(hDlg, dparam, en); EnableParam(hDlg, 0, dparam, en); } } } } } } static INT_PTR CALLBACK ParamDlgProc(HWND Window, UINT AMessage, WPARAM wParam, LPARAM lParam) { static bx_list_c *list = NULL; bx_param_c *param; bx_param_string_c *sparam; bx_list_c *tmplist; int cid; UINT_PTR code, id; UINT i, j, k; RECT r, r2; SIZE size; NMHDR nmhdr; char fname[BX_PATHNAME_LEN]; switch (AMessage) { case WM_CLOSE: cleanupDlgLists(); DestroyWindow(hwndTT); EndDialog(Window, -1); break; case WM_INITDIALOG: list = (bx_list_c*)SIM->get_param((const char*)lParam); SetWindowText(Window, list->get_title()); nextDlgID = 1; size = CreateParamList(Window, 0, 6, 6, FALSE, list); r.left = size.cx / 2 - 50; r.top = size.cy + 12; r.right = r.left + 50; r.bottom = r.top + 14; MapDialogRect(Window, &r); MoveWindow(GetDlgItem(Window, IDOK), r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, FALSE); r.left = size.cx / 2 + 10; r.top = size.cy + 12; r.right = r.left + 50; r.bottom = r.top + 14; MapDialogRect(Window, &r); MoveWindow(GetDlgItem(Window, IDCANCEL), r.left, r.top, r.right-r.left+1, r.bottom-r.top+1, FALSE); GetWindowRect(Window, &r2); r.left = 0; r.top = 0; r.right = size.cx + 18; r.bottom = size.cy + 52; MapDialogRect(Window, &r); MoveWindow(Window, r2.left, r2.top, r.right, r.bottom, TRUE); CreateParamDlgTooltip(Window); return TRUE; case WM_COMMAND: code = LOWORD(wParam); switch (code) { case IDCANCEL: cleanupDlgLists(); DestroyWindow(hwndTT); EndDialog(Window, -1); break; case IDOK: SetParamList(Window, list); cleanupDlgLists(); DestroyWindow(hwndTT); EndDialog(Window, 1); break; default: if ((code >= ID_BROWSE) && (code < (int)(ID_BROWSE + nextDlgID))) { i = (UINT)(code - ID_BROWSE); sparam = (bx_param_string_c *)findParamFromDlgID(i); if (sparam != NULL) { GetDlgItemText(Window, ID_PARAM + i, fname, BX_PATHNAME_LEN); if (AskFilename(Window, (bx_param_filename_c *)sparam, fname) > 0) { SetWindowText(GetDlgItem(Window, ID_PARAM + i), fname); SendMessage(GetDlgItem(Window, ID_PARAM + i), EM_SETMODIFY, 1, 0); SetFocus(GetDlgItem(Window, ID_PARAM + i)); } } } else if ((code >= ID_PARAM) && (code < (int)(ID_PARAM + nextDlgID))) { i = (UINT)(code - ID_PARAM); param = findParamFromDlgID(i); if (param != NULL) { ProcessDependentList(Window, param, TRUE); } } } break; case WM_NOTIFY: nmhdr = *(LPNMHDR)lParam; code = nmhdr.code; id = nmhdr.idFrom; if (code == (UINT)TCN_SELCHANGE) { j = (UINT)(id - ID_PARAM); tmplist = (bx_list_c *)findParamFromDlgID(j); if (tmplist != NULL) { k = (UINT)TabCtrl_GetCurSel(GetDlgItem(Window, id)); cid = findDlgListBaseID(tmplist); for (i = 0; i < (UINT)tmplist->get_size(); i++) { ShowParamList(Window, cid + i, (i == k), (bx_list_c*)tmplist->get(i)); } } } else if (code == (UINT)UDN_DELTAPOS) { if (id >= ID_UPDOWN) { PostMessage(GetDlgItem(Window, ID_PARAM + (id - ID_UPDOWN)), EM_SETMODIFY, TRUE, 0); } } else { OnWMNotify(lParam); } break; case WM_CTLCOLOREDIT: SetTextColor((HDC)wParam, GetSysColor(COLOR_WINDOWTEXT)); return (INT_PTR)GetSysColorBrush(COLOR_WINDOW); break; } return 0; } int win32ParamDialog(HWND parent, const char *menu) { INT_PTR ret; InitDlgFont(); ret = DialogBoxParam(NULL, MAKEINTRESOURCE(PARAM_DLG), parent, (DLGPROC)ParamDlgProc, (LPARAM)menu); DeleteObject(DlgFont); return ret; } #endif // BX_USE_TEXTCONFIG && defined(WIN32)