Update to v106r57 release.
byuu says:
I've added tool tips to hiro for Windows, GTK, and Qt. I'm unsure how to
add them for Cocoa. I wasted am embarrassing ~14 hours implementing tool
tips from scratch on Windows, because the `TOOLTIPS_CLASS` widget just
absolutely refused to show up, no matter what I tried. As such, they're
not quite 100% native, but I would really appreciate any patch
submissions to help improve my implementation.
I added tool tips to all of the confusing settings in bsnes. And of
course, for those of you who don't like them, there's a configuration
file setting to turn them off globally.
I also improved Mega Drive handling of the Game Genie a bit, and
restructured the way the Settings class works in bsnes.
Starting now, I'm feature-freezing bsnes and higan. From this point
forward:
- polishing up and fixing bugs caused by the ruby/hiro changes
- adding DRC to XAudio2, and maybe exclusive mode to WGL
- correcting FEoEZ (English) to load and work again out of the box
Once that's done, a final beta of bsnes will go out, I'll fix any
reported bugs that I'm able to, and then v107 should be ready. This time
with higan being functional, but marked as v107 beta. v108 will restore
higan to production status again, alongside bsnes.
2018-08-08 08:46:58 +00:00
|
|
|
namespace hiro {
|
|
|
|
|
|
|
|
static auto CALLBACK ToolTip_windowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) -> LRESULT {
|
|
|
|
if(auto toolTip = (pToolTip*)GetWindowLongPtr(hwnd, GWLP_USERDATA)) {
|
|
|
|
if(auto result = toolTip->windowProc(hwnd, msg, wparam, lparam)) {
|
|
|
|
return result();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return DefWindowProc(hwnd, msg, wparam, lparam);
|
|
|
|
}
|
|
|
|
|
|
|
|
pToolTip::pToolTip(const string& toolTipText) {
|
|
|
|
text = toolTipText;
|
|
|
|
|
|
|
|
htheme = OpenThemeData(hwnd, L"TOOLTIP");
|
|
|
|
hwnd = CreateWindowEx(
|
|
|
|
WS_EX_TOOLWINDOW | WS_EX_TOPMOST | (htheme ? WS_EX_LAYERED : 0), L"hiroToolTip", L"",
|
|
|
|
WS_POPUP, 0, 0, 0, 0,
|
|
|
|
0, 0, GetModuleHandle(0), 0
|
|
|
|
);
|
|
|
|
SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)this);
|
|
|
|
tracking.x = -1;
|
|
|
|
tracking.y = -1;
|
|
|
|
|
Update to 20180809 release.
byuu says:
The Windows port can now run the emulation while navigating menus,
moving windows, and resizing windows. The main window also doesn't try
so hard to constantly clear itself. This may leave a bit of unwelcome
residue behind in some video drivers during resize, but under most
drivers, it lets you resize without a huge amount of flickering.
On all platforms, I now also run the emulation during MessageWindow
modal events, where I didn't before.
I'm thinking we should probably mute the audio during modal periods,
since it can generate a good deal of distortion.
The tooltip timeout was increased to ten seconds.
On Windows, the enter key can now activate buttons, so you can more
quickly dismiss MessageDialog windows. This part may not actually work
... I'm in the middle of trying to get messages out of the global
`Application_windowProc` hook and into the individual `Widget_windowProc`
hooks, so I need to do some testing.
I fixed a bug where changing the input driver wouldn't immediately
reload the input/hotkey settings lists properly.
I also went from disabling the driver "Change" button when the currently
active driver is selected in the list, to instead setting it to say
"Reload", and I also added a tool tip to the input driver reload button,
advising that if you're using DirectInput or SDL, you can hit "Reload"
to rescan for hotplugged gamepads without needing to restart the
emulator. XInput and udev have auto hotswap support. If we can ever get
that into DirectInput and SDL, then I'll remove the tooltip. But
regardless, the reload functionality is nice to have for all drivers.
I'm not sure what should happen when a user changes their driver
selection while a game is loaded, gets the warning dialog, chooses not
to change it, and then closes the emulator. Currently, it will make the
change happen the next time you start the emulator. This feels a bit
unexpected, but when you change the selection without a game loaded, it
takes immediate effect. So I'm not really sure what's best here.
2018-08-10 05:02:59 +00:00
|
|
|
timeout.setInterval(Timeout);
|
Update to v106r57 release.
byuu says:
I've added tool tips to hiro for Windows, GTK, and Qt. I'm unsure how to
add them for Cocoa. I wasted am embarrassing ~14 hours implementing tool
tips from scratch on Windows, because the `TOOLTIPS_CLASS` widget just
absolutely refused to show up, no matter what I tried. As such, they're
not quite 100% native, but I would really appreciate any patch
submissions to help improve my implementation.
I added tool tips to all of the confusing settings in bsnes. And of
course, for those of you who don't like them, there's a configuration
file setting to turn them off globally.
I also improved Mega Drive handling of the Game Genie a bit, and
restructured the way the Settings class works in bsnes.
Starting now, I'm feature-freezing bsnes and higan. From this point
forward:
- polishing up and fixing bugs caused by the ruby/hiro changes
- adding DRC to XAudio2, and maybe exclusive mode to WGL
- correcting FEoEZ (English) to load and work again out of the box
Once that's done, a final beta of bsnes will go out, I'll fix any
reported bugs that I'm able to, and then v107 should be ready. This time
with higan being functional, but marked as v107 beta. v108 will restore
higan to production status again, alongside bsnes.
2018-08-08 08:46:58 +00:00
|
|
|
timeout.onActivate([&] { hide(); });
|
|
|
|
}
|
|
|
|
|
|
|
|
pToolTip::~pToolTip() {
|
|
|
|
hide();
|
|
|
|
if(htheme) { CloseThemeData(htheme); htheme = nullptr; }
|
|
|
|
if(hwnd) { DestroyWindow(hwnd); hwnd = nullptr; }
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pToolTip::drawLayered() -> void {
|
|
|
|
auto hdcOutput = GetDC(nullptr);
|
|
|
|
|
|
|
|
uint32_t* below = nullptr;
|
|
|
|
auto hdcBelow = CreateCompatibleDC(hdcOutput);
|
|
|
|
auto hbmBelow = CreateBitmap(hdcBelow, size.cx, size.cy, below);
|
|
|
|
SelectObject(hdcBelow, hbmBelow);
|
|
|
|
RECT rc{};
|
|
|
|
rc.left = 0, rc.top = 0, rc.right = size.cx, rc.bottom = size.cy;
|
|
|
|
DrawThemeBackground(htheme, hdcBelow, TTP_STANDARD, TTSS_NORMAL, &rc, nullptr);
|
|
|
|
|
|
|
|
uint32_t* above = nullptr;
|
|
|
|
auto hdcAbove = CreateCompatibleDC(hdcOutput);
|
|
|
|
auto hbmAbove = CreateBitmap(hdcAbove, size.cx, size.cy, above);
|
|
|
|
SelectObject(hdcAbove, hbmAbove);
|
|
|
|
|
|
|
|
memory::copy<uint32_t>(above, below, size.cx * size.cy);
|
|
|
|
|
|
|
|
auto hfont = pFont::create(Font());
|
|
|
|
SelectObject(hdcAbove, hfont);
|
|
|
|
SetBkMode(hdcAbove, TRANSPARENT);
|
|
|
|
SetTextColor(hdcAbove, RGB(0, 0, 0));
|
|
|
|
utf16_t drawText(text);
|
|
|
|
rc.left += 6, rc.top += 6, rc.right -= 6, rc.bottom -= 6;
|
|
|
|
DrawText(hdcAbove, drawText, -1, &rc, DT_LEFT | DT_TOP);
|
|
|
|
DeleteObject(hfont);
|
|
|
|
|
|
|
|
for(uint n : range(size.cx * size.cy)) {
|
|
|
|
below[n] = (below[n] & 0xff000000) | (above[n] & 0x00ffffff);
|
|
|
|
}
|
|
|
|
|
|
|
|
BLENDFUNCTION blend{};
|
|
|
|
blend.BlendOp = AC_SRC_OVER;
|
|
|
|
blend.SourceConstantAlpha = 255;
|
|
|
|
blend.AlphaFormat = AC_SRC_ALPHA;
|
|
|
|
POINT zero{};
|
|
|
|
zero.x = 0, zero.y = 0;
|
|
|
|
UpdateLayeredWindow(hwnd, hdcOutput, &position, &size, hdcBelow, &zero, RGB(0, 0, 0), &blend, ULW_ALPHA);
|
|
|
|
|
|
|
|
DeleteObject(hbmAbove);
|
|
|
|
DeleteObject(hbmBelow);
|
|
|
|
DeleteDC(hdcAbove);
|
|
|
|
DeleteDC(hdcBelow);
|
|
|
|
ReleaseDC(nullptr, hdcOutput);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pToolTip::drawOpaque() -> void {
|
|
|
|
PAINTSTRUCT ps;
|
|
|
|
BeginPaint(hwnd, &ps);
|
|
|
|
RECT rc{};
|
|
|
|
GetClientRect(hwnd, &rc);
|
|
|
|
auto brush = CreateSolidBrush(RGB(0, 0, 0));
|
|
|
|
FillRect(ps.hdc, &rc, brush);
|
|
|
|
DeleteObject(brush);
|
|
|
|
rc.left += 1, rc.top += 1, rc.right -= 1, rc.bottom -= 1;
|
|
|
|
brush = CreateSolidBrush(RGB(255, 255, 225));
|
|
|
|
FillRect(ps.hdc, &rc, brush);
|
|
|
|
DeleteObject(brush);
|
|
|
|
rc.left += 5, rc.top += 5, rc.right -= 5, rc.bottom -= 5;
|
|
|
|
SetBkMode(ps.hdc, TRANSPARENT);
|
|
|
|
auto font = pFont::create(Font());
|
|
|
|
SelectObject(ps.hdc, font);
|
|
|
|
SetTextColor(ps.hdc, RGB(0, 0, 0));
|
|
|
|
DrawText(ps.hdc, utf16_t(text), -1, &rc, DT_LEFT | DT_TOP);
|
|
|
|
DeleteObject(font);
|
|
|
|
EndPaint(hwnd, &ps);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pToolTip::show() -> void {
|
|
|
|
if(auto toolTip = pApplication::state().toolTip) {
|
|
|
|
if(toolTip != this) toolTip->hide();
|
|
|
|
}
|
|
|
|
pApplication::state().toolTip = this;
|
|
|
|
|
|
|
|
GetCursorPos(&position);
|
|
|
|
if(position.x == tracking.x && position.y == tracking.y) return;
|
|
|
|
tracking.x = position.x, tracking.y = position.y;
|
|
|
|
|
|
|
|
position.y += 18;
|
|
|
|
auto textSize = pFont::size(Font(), text ? text : " ");
|
|
|
|
size.cx = 12 + textSize.width();
|
|
|
|
size.cy = 12 + textSize.height();
|
|
|
|
|
|
|
|
//try to keep the tool-tip onscreen
|
|
|
|
auto desktop = pDesktop::size();
|
|
|
|
if(position.x + size.cx >= desktop.width ()) position.x = desktop.width () - size.cx;
|
|
|
|
if(position.y + size.cy >= desktop.height()) position.y = desktop.height() - size.cy;
|
|
|
|
if(position.x < 0) position.x = 0;
|
|
|
|
if(position.y < 0) position.y = 0;
|
|
|
|
|
|
|
|
SetWindowPos(hwnd, HWND_TOP, position.x, position.y, size.cx, size.cy, SWP_NOACTIVATE | SWP_SHOWWINDOW);
|
|
|
|
if(htheme) drawLayered();
|
|
|
|
|
|
|
|
timeout.setEnabled(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pToolTip::hide() -> void {
|
|
|
|
pApplication::state().toolTip = nullptr;
|
|
|
|
timeout.setEnabled(false);
|
|
|
|
ShowWindow(hwnd, SW_HIDE);
|
|
|
|
GetCursorPos(&tracking);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pToolTip::windowProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) -> maybe<LRESULT> {
|
|
|
|
switch(msg) {
|
|
|
|
case WM_ERASEBKGND:
|
|
|
|
case WM_PAINT:
|
|
|
|
if(htheme) break;
|
|
|
|
drawOpaque();
|
|
|
|
return msg == WM_ERASEBKGND;
|
|
|
|
case WM_MOUSEMOVE:
|
|
|
|
case WM_MOUSELEAVE: {
|
|
|
|
POINT point{};
|
|
|
|
GetCursorPos(&point);
|
|
|
|
if(point.x != tracking.x || point.y != tracking.y) hide();
|
|
|
|
} break;
|
|
|
|
case WM_LBUTTONDOWN: case WM_LBUTTONUP:
|
|
|
|
case WM_MBUTTONDOWN: case WM_MBUTTONUP:
|
|
|
|
case WM_RBUTTONDOWN: case WM_RBUTTONUP:
|
|
|
|
hide();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|