mirror of
https://github.com/bsnes-emu/bsnes.git
synced 2025-04-02 10:42:14 -04:00
byuu says: Changelog: - fixed major nall/vector/prepend bug - renamed hiro/ListView to hiro/TableView - added new hiro/ListView control which is a simplified abstraction of hiro/TableView - updated higan's cheat database window and icarus' scan dialog to use the new ListView control - compilation works once again on all platforms (Windows, Cocoa, GTK, Qt) - the loki skeleton compiles once again (removed nall/DSP references; updated port/device ID names) Small catch: need to capture layout resize events internally in Windows to call resizeColumns. For now, just resize the icarus window to get it to use the full window width for list view items.
397 lines
13 KiB
C++
397 lines
13 KiB
C++
namespace hiro {
|
|
|
|
static const uint Windows2000 = 0x0500;
|
|
static const uint WindowsXP = 0x0501;
|
|
static const uint WindowsVista = 0x0600;
|
|
static const uint Windows7 = 0x0601;
|
|
|
|
static auto Button_CustomDraw(HWND, PAINTSTRUCT&, bool, bool, bool, unsigned, const Font&, const image&, Orientation, const string&) -> void;
|
|
|
|
static auto OsVersion() -> unsigned {
|
|
OSVERSIONINFO versionInfo{0};
|
|
versionInfo.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
|
|
GetVersionEx(&versionInfo);
|
|
return (versionInfo.dwMajorVersion << 8) + (versionInfo.dwMajorVersion << 0);
|
|
}
|
|
|
|
static auto CreateBitmap(image icon) -> HBITMAP {
|
|
icon.alphaMultiply(); //Windows AlphaBlend() requires premultiplied image data
|
|
icon.transform();
|
|
HDC hdc = GetDC(0);
|
|
BITMAPINFO bitmapInfo;
|
|
memset(&bitmapInfo, 0, sizeof(BITMAPINFO));
|
|
bitmapInfo.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
|
bitmapInfo.bmiHeader.biWidth = icon.width();
|
|
bitmapInfo.bmiHeader.biHeight = -(signed)icon.height(); //bitmaps are stored upside down unless we negate height
|
|
bitmapInfo.bmiHeader.biPlanes = 1;
|
|
bitmapInfo.bmiHeader.biBitCount = 32;
|
|
bitmapInfo.bmiHeader.biCompression = BI_RGB;
|
|
bitmapInfo.bmiHeader.biSizeImage = icon.size();
|
|
void* bits = nullptr;
|
|
HBITMAP hbitmap = CreateDIBSection(hdc, &bitmapInfo, DIB_RGB_COLORS, &bits, NULL, 0);
|
|
if(bits) memory::copy(bits, icon.data(), icon.size());
|
|
ReleaseDC(0, hdc);
|
|
return hbitmap;
|
|
}
|
|
|
|
static auto CreateRGB(const Color& color) -> COLORREF {
|
|
return RGB(color.red(), color.green(), color.blue());
|
|
}
|
|
|
|
static auto DropPaths(WPARAM wparam) -> lstring {
|
|
auto dropList = HDROP(wparam);
|
|
auto fileCount = DragQueryFile(dropList, ~0u, nullptr, 0);
|
|
|
|
lstring paths;
|
|
for(auto n : range(fileCount)) {
|
|
auto length = DragQueryFile(dropList, n, nullptr, 0);
|
|
auto buffer = new wchar_t[length + 1];
|
|
|
|
if(DragQueryFile(dropList, n, buffer, length + 1)) {
|
|
string path = (const char*)utf8_t(buffer);
|
|
path.transform("\\", "/");
|
|
if(directory::exists(path) && !path.endsWith("/")) path.append("/");
|
|
paths.append(path);
|
|
}
|
|
|
|
delete[] buffer;
|
|
}
|
|
|
|
return paths;
|
|
}
|
|
|
|
static auto GetWindowZOrder(HWND hwnd) -> unsigned {
|
|
unsigned z = 0;
|
|
for(HWND next = hwnd; next != NULL; next = GetWindow(next, GW_HWNDPREV)) z++;
|
|
return z;
|
|
}
|
|
|
|
static auto ImageList_Append(HIMAGELIST imageList, image icon, unsigned scale) -> void {
|
|
if(icon) {
|
|
icon.scale(scale, scale);
|
|
} else {
|
|
icon.allocate(scale, scale);
|
|
icon.fill(GetSysColor(COLOR_WINDOW));
|
|
}
|
|
HBITMAP bitmap = CreateBitmap(icon);
|
|
ImageList_Add(imageList, bitmap, nullptr);
|
|
DeleteObject(bitmap);
|
|
}
|
|
|
|
//post message only if said message is not already pending in the queue
|
|
static auto PostMessageOnce(HWND hwnd, UINT id, WPARAM wparam, LPARAM lparam) -> void {
|
|
MSG msg;
|
|
if(!PeekMessage(&msg, hwnd, id, id, PM_NOREMOVE)) {
|
|
PostMessage(hwnd, id, wparam, lparam);
|
|
}
|
|
}
|
|
|
|
static auto ScrollEvent(HWND hwnd, WPARAM wparam) -> unsigned {
|
|
SCROLLINFO info;
|
|
memset(&info, 0, sizeof(SCROLLINFO));
|
|
info.cbSize = sizeof(SCROLLINFO);
|
|
info.fMask = SIF_ALL;
|
|
GetScrollInfo(hwnd, SB_CTL, &info);
|
|
|
|
switch(LOWORD(wparam)) {
|
|
case SB_LEFT: info.nPos = info.nMin; break;
|
|
case SB_RIGHT: info.nPos = info.nMax; break;
|
|
case SB_LINELEFT: info.nPos--; break;
|
|
case SB_LINERIGHT: info.nPos++; break;
|
|
case SB_PAGELEFT: info.nPos -= info.nMax >> 3; break;
|
|
case SB_PAGERIGHT: info.nPos += info.nMax >> 3; break;
|
|
case SB_THUMBTRACK: info.nPos = info.nTrackPos; break;
|
|
}
|
|
|
|
info.fMask = SIF_POS;
|
|
SetScrollInfo(hwnd, SB_CTL, &info, TRUE);
|
|
|
|
//Windows may clamp position to scrollbar range
|
|
GetScrollInfo(hwnd, SB_CTL, &info);
|
|
return info.nPos;
|
|
}
|
|
|
|
//separate because PopupMenu HWND does not contain GWLP_USERDATA pointing at Window needed for Shared_windowProc
|
|
static auto CALLBACK Menu_windowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) -> LRESULT {
|
|
switch(msg) {
|
|
case WM_MENUCOMMAND: {
|
|
MENUITEMINFO mii{sizeof(MENUITEMINFO)};
|
|
mii.fMask = MIIM_DATA;
|
|
GetMenuItemInfo((HMENU)lparam, wparam, true, &mii);
|
|
|
|
auto object = (mObject*)mii.dwItemData;
|
|
if(!object) break;
|
|
|
|
#if defined(Hiro_MenuItem)
|
|
if(auto menuItem = dynamic_cast<mMenuItem*>(object)) {
|
|
return menuItem->self()->onActivate(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_MenuCheckItem)
|
|
if(auto menuCheckItem = dynamic_cast<mMenuCheckItem*>(object)) {
|
|
return menuCheckItem->self()->onToggle(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_MenuRadioItem)
|
|
if(auto menuRadioItem = dynamic_cast<mMenuRadioItem*>(object)) {
|
|
return menuRadioItem->self()->onActivate(), false;
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
}
|
|
|
|
static auto CALLBACK Shared_windowProc(WindowProc windowProc, HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) -> LRESULT {
|
|
if(Application::state.quit) return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
|
|
auto object = (mObject*)GetWindowLongPtr(hwnd, GWLP_USERDATA);
|
|
if(!object) return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
auto window = dynamic_cast<mWindow*>(object);
|
|
if(!window) window = object->parentWindow(true);
|
|
if(!window) return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
auto pWindow = window->self();
|
|
if(!pWindow) return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
|
|
if(pWindow->_modalityDisabled()) return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
|
|
switch(msg) {
|
|
case WM_CTLCOLORBTN:
|
|
case WM_CTLCOLOREDIT:
|
|
case WM_CTLCOLORSTATIC: {
|
|
auto object = (mObject*)GetWindowLongPtr((HWND)lparam, GWLP_USERDATA);
|
|
if(!object) break;
|
|
|
|
//allow custom colors for various widgets
|
|
//note that this happens always: default colors are black text on a white background, unless overridden
|
|
//this intentionally overrides the default behavior of Windows to paint disabled controls with the window background color
|
|
|
|
#if defined(Hiro_Window) && defined(Hiro_TabFrame)
|
|
if(!object->parentTabFrame(true) && window->self()->hbrush) {
|
|
SetBkColor((HDC)wparam, window->self()->hbrushColor);
|
|
return (LRESULT)window->self()->hbrush;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_HexEdit)
|
|
if(auto hexEdit = dynamic_cast<mHexEdit*>(object)) {
|
|
if(auto background = hexEdit->backgroundColor()) SetBkColor((HDC)wparam, CreateRGB(background));
|
|
if(auto foreground = hexEdit->foregroundColor()) SetTextColor((HDC)wparam, CreateRGB(foreground));
|
|
return (LRESULT)hexEdit->self()->backgroundBrush;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_LineEdit)
|
|
if(auto lineEdit = dynamic_cast<mLineEdit*>(object)) {
|
|
if(auto background = lineEdit->backgroundColor()) SetBkColor((HDC)wparam, CreateRGB(background));
|
|
if(auto foreground = lineEdit->foregroundColor()) SetTextColor((HDC)wparam, CreateRGB(foreground));
|
|
return (LRESULT)lineEdit->self()->backgroundBrush;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_TextEdit)
|
|
if(auto textEdit = dynamic_cast<mTextEdit*>(object)) {
|
|
if(auto background = textEdit->backgroundColor()) SetBkColor((HDC)wparam, CreateRGB(background));
|
|
if(auto foreground = textEdit->foregroundColor()) SetTextColor((HDC)wparam, CreateRGB(foreground));
|
|
return (LRESULT)textEdit->self()->backgroundBrush;
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
|
|
case WM_DRAWITEM: {
|
|
auto drawItem = (LPDRAWITEMSTRUCT)lparam;
|
|
auto object = (mObject*)GetWindowLongPtr((HWND)drawItem->hwndItem, GWLP_USERDATA);
|
|
if(!object) break;
|
|
|
|
#if defined(Hiro_TabFrame)
|
|
if(auto tabFrame = dynamic_cast<mTabFrame*>(object)) {
|
|
return tabFrame->self()->onDrawItem(lparam), true;
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
|
|
case WM_MENUCOMMAND: {
|
|
return Menu_windowProc(hwnd, msg, wparam, lparam);
|
|
}
|
|
|
|
case WM_COMMAND: {
|
|
if(!lparam) break;
|
|
auto object = (mObject*)GetWindowLongPtr((HWND)lparam, GWLP_USERDATA);
|
|
if(!object) break;
|
|
|
|
#if defined(Hiro_Button)
|
|
if(auto button = dynamic_cast<mButton*>(object)) {
|
|
return button->self()->onActivate(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_CheckButton)
|
|
if(auto checkButton = dynamic_cast<mCheckButton*>(object)) {
|
|
return checkButton->self()->onToggle(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_CheckLabel)
|
|
if(auto checkLabel = dynamic_cast<mCheckLabel*>(object)) {
|
|
return checkLabel->self()->onToggle(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_ComboButton)
|
|
if(auto comboButton = dynamic_cast<mComboButton*>(object)) {
|
|
if(HIWORD(wparam) == CBN_SELCHANGE) {
|
|
return comboButton->self()->onChange(), false;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_LineEdit)
|
|
if(auto lineEdit = dynamic_cast<mLineEdit*>(object)) {
|
|
if(HIWORD(wparam) == EN_CHANGE) {
|
|
return lineEdit->self()->onChange(), false;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_RadioButton)
|
|
if(auto radioButton = dynamic_cast<mRadioButton*>(object)) {
|
|
return radioButton->self()->onActivate(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_RadioLabel)
|
|
if(auto radioLabel = dynamic_cast<mRadioLabel*>(object)) {
|
|
return radioLabel->self()->onActivate(), false;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_TextEdit)
|
|
if(auto textEdit = dynamic_cast<mTextEdit*>(object)) {
|
|
if(HIWORD(wparam) == EN_CHANGE) {
|
|
return textEdit->self()->onChange(), false;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
|
|
case WM_NOTIFY: {
|
|
//Widgets inside a TabFrame must be parented to it rather than the Window.
|
|
//This is critical for proper inheritance of styles and message passing.
|
|
//However, by doing this, some WM_NOTIFY messages end up being sent to both
|
|
//the TabFrame and the Window; while others are only sent to the TabFrame.
|
|
//To save code, hiro uses a shared callback for both of these cases.
|
|
//So when a message is sent to both, we ignore the TabFrame message.
|
|
bool isWindowCallback = (object == window);
|
|
|
|
auto header = (LPNMHDR)lparam;
|
|
auto object = (mObject*)GetWindowLongPtr((HWND)header->hwndFrom, GWLP_USERDATA);
|
|
if(!object) break;
|
|
|
|
#if defined(Hiro_TableView)
|
|
if(auto tableView = dynamic_cast<mTableView*>(object)) {
|
|
if(header->code == LVN_ITEMACTIVATE) {
|
|
tableView->self()->onActivate(lparam);
|
|
break;
|
|
}
|
|
if(header->code == LVN_ITEMCHANGED) {
|
|
tableView->self()->onChange(lparam);
|
|
break;
|
|
}
|
|
if(header->code == LVN_COLUMNCLICK) {
|
|
if(isWindowCallback) tableView->self()->onSort(lparam);
|
|
break;
|
|
}
|
|
if(header->code == NM_CLICK || header->code == NM_DBLCLK) {
|
|
//onToggle performs the test to ensure the TableViewItem clicked was checkable
|
|
if(isWindowCallback) tableView->self()->onToggle(lparam);
|
|
break;
|
|
}
|
|
if(header->code == NM_RCLICK) {
|
|
if(isWindowCallback) tableView->self()->onContext(lparam);
|
|
break;
|
|
}
|
|
if(header->code == NM_CUSTOMDRAW) {
|
|
return tableView->self()->onCustomDraw(lparam);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_TabFrame)
|
|
if(auto tabFrame = dynamic_cast<mTabFrame*>(object)) {
|
|
if(header->code == TCN_SELCHANGE) {
|
|
tabFrame->self()->onChange();
|
|
break;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
|
|
#if defined(Hiro_TableView)
|
|
case AppMessage::TableView_doPaint: {
|
|
if(auto tableView = (mTableView*)lparam) {
|
|
if(auto self = tableView->self()) InvalidateRect(self->hwnd, nullptr, true);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case AppMessage::TableView_onActivate: {
|
|
if(auto tableView = (mTableView*)lparam) tableView->doActivate();
|
|
break;
|
|
}
|
|
|
|
case AppMessage::TableView_onChange: {
|
|
if(auto tableView = (mTableView*)lparam) tableView->doChange();
|
|
}
|
|
#endif
|
|
|
|
case WM_HSCROLL:
|
|
case WM_VSCROLL: {
|
|
if(!lparam) break;
|
|
auto object = (mObject*)GetWindowLongPtr((HWND)lparam, GWLP_USERDATA);
|
|
if(!object) break;
|
|
|
|
#if defined(Hiro_HorizontalScrollBar)
|
|
if(auto horizontalScrollBar = dynamic_cast<mHorizontalScrollBar*>(object)) {
|
|
return horizontalScrollBar->self()->onChange(wparam), true;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_HorizontalSlider)
|
|
if(auto horizontalSlider = dynamic_cast<mHorizontalSlider*>(object)) {
|
|
return horizontalSlider->self()->onChange(), true;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_VerticalScrollBar)
|
|
if(auto verticalScrollBar = dynamic_cast<mVerticalScrollBar*>(object)) {
|
|
return verticalScrollBar->self()->onChange(wparam), true;
|
|
}
|
|
#endif
|
|
|
|
#if defined(Hiro_VerticalSlider)
|
|
if(auto verticalSlider = dynamic_cast<mVerticalSlider*>(object)) {
|
|
return verticalSlider->self()->onChange(), true;
|
|
}
|
|
#endif
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
return windowProc(hwnd, msg, wparam, lparam);
|
|
}
|
|
|
|
}
|