5744 lines
127 KiB
C++
5744 lines
127 KiB
C++
#include <precomp.h>
|
|
|
|
#include <bfc/wasabi_std.h>
|
|
#include <bfc/wasabi_std_wnd.h>
|
|
#include <api/wnd/wndevent.h>
|
|
|
|
#include <bfc/bfc_assert.h>
|
|
#include <api/wnd/wndclass/tooltip.h>
|
|
#include <api/wnd/cursor.h>
|
|
#include <api/wnd/accessible.h>
|
|
#include <api/service/svcs/svc_accessibility.h>
|
|
#include <api/wnd/paintsets.h>
|
|
#include <api/wnd/PaintCanvas.h>
|
|
|
|
#ifdef _WIN32
|
|
#include <shellapi.h> // for HDROP
|
|
#endif
|
|
#include <tataki/canvas/bltcanvas.h>
|
|
|
|
#define DESKTOPALPHA
|
|
#define REFRESH_RATE 25
|
|
#define DRAWTIMERID 125
|
|
|
|
#include <api/wnd/basewnd.h>
|
|
#include <api/wnd/usermsg.h>
|
|
|
|
#include <api/wnd/paintcb.h>
|
|
#include <tataki/canvas/canvas.h>
|
|
#include <bfc/file/filename.h>
|
|
#include <tataki/region/region.h>
|
|
#include <api/wnd/wndclass/guiobjwnd.h>
|
|
#include <api/script/scriptguid.h>
|
|
#include <api/wnd/notifmsg.h>
|
|
#include <api/metrics/metricscb.h>
|
|
|
|
#include <api/wndmgr/gc.h>
|
|
#include <api/wndmgr/layout.h>
|
|
|
|
namespace Agave
|
|
{
|
|
#include "../Agave/Config/api_config.h"
|
|
}
|
|
|
|
|
|
|
|
//#define TIP_TIMER_ID 1601
|
|
#define TIP_DESTROYTIMER_ID 1602
|
|
#define TIP_AWAY_ID 1603
|
|
#define TIP_AWAY_DELAY 100
|
|
|
|
#define TIP_TIMER_THRESHOLD 350
|
|
#define TIP_LENGTH 3000
|
|
|
|
#define VCHILD_TIMER_ID_MIN 2000
|
|
#define VCHILD_TIMER_ID_MAX 2100
|
|
|
|
#define BUFFEREDMSG_TIMER_ID 1604
|
|
|
|
#define DEFERREDCB_INVALIDATE 0x201 // move to .h
|
|
#define DEFERREDCB_FOCUSFIRST 0x202 // move to .h
|
|
#define DC_KILLGHOST 0x204
|
|
|
|
#ifdef _WIN32
|
|
#define WM_DEFER_CALLBACK (WM_USER+0x333)
|
|
#endif
|
|
class DragSet : public PtrList<void>, public NamedW {};
|
|
|
|
//CUT? static void register_wndClass(HINSTANCE);
|
|
|
|
//CUT? #define ROOTSTRING "RootWnd"
|
|
|
|
//CUT? #define BASEWNDCLASSNAME "BaseWindow_" ROOTSTRING
|
|
|
|
#ifndef WM_MOUSEWHEEL
|
|
#define WM_MOUSEWHEEL 0x20A
|
|
#endif
|
|
|
|
static ifc_window *stickyWnd;
|
|
static RECT sticky;
|
|
|
|
static UINT WINAMP_WM_DIRECT_MOUSE_WHEEL = WM_NULL;
|
|
|
|
/*api_window *api_window::rootwndFromPoint(POINT &point, int level) {
|
|
api_window *wnd;
|
|
wnd = WASABI_API_WND->rootWndFromPoint(&point);
|
|
return api_window::rootwndFromRootWnd(wnd, level, &point);
|
|
}
|
|
|
|
api_window *api_window::rootwndFromRootWnd(api_window *wnd, int level, POINT *point) {
|
|
|
|
for (;;) {
|
|
if (wnd == NULL || level < 0) return NULL;
|
|
if (point) {
|
|
RECT r;
|
|
wnd->getWindowRect(&r);
|
|
if (!PtInRect(&r, *point)) return NULL; // PORT ME
|
|
}
|
|
if (level == 0) return wnd;
|
|
wnd = wnd->getRootWndParent();
|
|
level--;
|
|
}
|
|
// should never get here
|
|
}*/
|
|
|
|
static BOOL DisabledWindow_OnMouseClick(HWND hwnd)
|
|
{
|
|
DWORD windowStyle = (DWORD)GetWindowLongPtrW(hwnd, GWL_STYLE);
|
|
if (WS_DISABLED != ((WS_CHILD | WS_DISABLED) & windowStyle))
|
|
return FALSE;
|
|
|
|
HWND hActive = GetActiveWindow();
|
|
HWND hPopup = GetWindow(hwnd, GW_ENABLEDPOPUP);
|
|
|
|
BOOL beepOk = (hPopup == hActive || hwnd == GetWindow(hActive, GW_OWNER));
|
|
if (!beepOk && NULL == hPopup)
|
|
{
|
|
for (HWND hWalker = hwnd; ;)
|
|
{
|
|
hWalker = GetWindow(hWalker, GW_OWNER);
|
|
if (NULL == hWalker || (0 != (WS_CHILD & GetWindowLongPtrW(hWalker, GWL_STYLE))))
|
|
break;
|
|
if (hActive == GetWindow(hWalker, GW_ENABLEDPOPUP))
|
|
{
|
|
beepOk = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (beepOk)
|
|
{
|
|
static const GUID accessibilityConfigGroupGUID =
|
|
{ 0xe2e7f4a, 0x7c51, 0x478f, { 0x87, 0x74, 0xab, 0xbc, 0xf6, 0xd5, 0xa8, 0x57 } };
|
|
|
|
#define GetBoolConfig(__group, __itemName, __default)\
|
|
((NULL != (__group)) && NULL != (item = group->GetItem(__itemName)) ? item->GetBool() : (__default))
|
|
|
|
waServiceFactory *serviceFactory = WASABI_API_SVC->service_getServiceByGuid(Agave::AgaveConfigGUID);
|
|
Agave::api_config *config = (NULL != serviceFactory) ? (Agave::api_config *)serviceFactory->getInterface() : NULL;
|
|
Agave::ifc_configgroup *group = (NULL != config) ? config->GetGroup(accessibilityConfigGroupGUID) : NULL;
|
|
Agave::ifc_configitem *item;
|
|
|
|
if (GetBoolConfig(group, L"modalflash", true))
|
|
{
|
|
FLASHWINFO flashInfo;
|
|
flashInfo.cbSize = sizeof(FLASHWINFO);
|
|
flashInfo.hwnd = hActive;
|
|
flashInfo.dwFlags = FLASHW_CAPTION;
|
|
flashInfo.uCount = 2;
|
|
flashInfo.dwTimeout = 100;
|
|
FlashWindowEx(&flashInfo);
|
|
}
|
|
|
|
if (GetBoolConfig(group, L"modalbeep", false))
|
|
MessageBeep(MB_OK);
|
|
|
|
if (NULL != config)
|
|
serviceFactory->releaseInterface(config);
|
|
}
|
|
else
|
|
{
|
|
for (HWND hWalker = hwnd; NULL == hPopup;)
|
|
{
|
|
hWalker = GetWindow(hWalker, GW_OWNER);
|
|
if (NULL == hWalker || (0 != (WS_CHILD & GetWindowLongPtrW(hWalker, GWL_STYLE))))
|
|
break;
|
|
hPopup = GetWindow(hWalker, GW_ENABLEDPOPUP);
|
|
}
|
|
|
|
SetWindowPos(hwnd, HWND_TOP, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOSIZE | SWP_NOMOVE);
|
|
if (NULL != hPopup && hPopup != hwnd)
|
|
{
|
|
BringWindowToTop(hPopup);
|
|
SetActiveWindow(hPopup);
|
|
}
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
int WndWatcher::viewer_onItemDeleted(ifc_dependent *item)
|
|
{
|
|
if (item == dep)
|
|
{
|
|
dep = NULL;
|
|
watcher->wndwatcher_onDeleteWindow(watched);
|
|
watched = NULL;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
BaseWnd::BaseWnd()
|
|
{
|
|
uiwaslocked = 0;
|
|
m_takenOver = 0;
|
|
rootfocus = NULL;
|
|
rootfocuswatcher.setWatcher(this);
|
|
alwaysontop = 0;
|
|
customdefaultcursor = NULL;
|
|
preventcancelcapture = 0;
|
|
ratiolinked = 1;
|
|
deleting = 0;
|
|
hinstance = NULL;
|
|
hwnd = NULL;
|
|
parentWnd = NULL;
|
|
dragging = 0;
|
|
prevtarg = NULL;
|
|
inputCaptured = 0;
|
|
btexture = NULL;
|
|
postoninit = 0;
|
|
inited = 0;
|
|
skipnextfocus = 0;
|
|
ncb = FALSE;
|
|
accessible = NULL;
|
|
|
|
tooltip = NULL;
|
|
tip_done = FALSE;
|
|
tipshowtimer = FALSE;
|
|
tipawaytimer = FALSE;
|
|
tipdestroytimer = FALSE;
|
|
start_hidden = 0;
|
|
notifyWindow = NULL;
|
|
lastClick[0] = 0;
|
|
lastClick[1] = 0;
|
|
lastClickP[0].x = 0;
|
|
lastClickP[0].y = 0;
|
|
lastClickP[1].x = 0;
|
|
lastClickP[1].y = 0;
|
|
destroying = FALSE;
|
|
|
|
curVirtualChildCaptured = NULL;
|
|
curVirtualChildFocus = NULL;
|
|
|
|
virtualCanvas = NULL; virtualCanvasH = virtualCanvasW = 0;
|
|
deferedInvalidRgn = NULL;
|
|
|
|
hasfocus = 0;
|
|
focus_on_click = 1;
|
|
lastnullregion = 0;
|
|
ratio = 1;
|
|
lastratio = 1;
|
|
rwidth = rheight = 0;
|
|
skin_id = -1;
|
|
wndalpha = 255;
|
|
activealpha = 255;
|
|
inactivealpha = 255;
|
|
w2k_alpha = 0; //FUCKO
|
|
scalecanvas = NULL;
|
|
clickthrough = 0;
|
|
|
|
mustquit = 0;
|
|
returnvalue = 0;
|
|
notifyid = 0;
|
|
cloaked = 0;
|
|
disable_tooltip_til_recapture = 0;
|
|
|
|
subtractorrgn = NULL;
|
|
composedrgn = NULL;
|
|
wndregioninvalid = 1;
|
|
regionop = REGIONOP_NONE;
|
|
rectrgn = 1;
|
|
need_flush_cascaderepaint = 0;
|
|
deferedCascadeRepaintRgn = NULL;
|
|
this_visible = 0;
|
|
this_enabled = 1;
|
|
renderbasetexture = 0;
|
|
oldCapture = NULL;
|
|
my_guiobject = NULL;
|
|
want_autoresize_after_init = 0;
|
|
resizecount = 0;
|
|
suggested_w = 320;
|
|
suggested_h = 200;
|
|
maximum_w = maximum_h = AUTOWH;
|
|
minimum_w = minimum_h = AUTOWH;
|
|
rx = 0;
|
|
ry = 0;
|
|
rwidth = 0;
|
|
rheight = 0;
|
|
allow_deactivate = 1;
|
|
minimized = 0;
|
|
inonresize = 0;
|
|
#ifndef WA3COMPATIBILITY
|
|
m_target = NULL;
|
|
#endif
|
|
|
|
nodoubleclick = noleftclick = norightclick = nomousemove = nocontextmnu = 0;
|
|
focusEventsEnabled = 1;
|
|
maximized = 0;
|
|
MEMSET(&restore_rect, 0, sizeof(RECT));
|
|
ghostbust = 0;
|
|
|
|
lastActiveWnd = NULL;
|
|
}
|
|
|
|
BaseWnd::~BaseWnd()
|
|
{
|
|
//ASSERT(virtualChildren.getNumItems() == 0);
|
|
childtabs.deleteAll();
|
|
if (WASABI_API_WND && WASABI_API_WND->getModalWnd() == this) WASABI_API_WND->popModalWnd(this);
|
|
destroying = TRUE;
|
|
curVirtualChildFocus = NULL;
|
|
#ifdef _WIN32
|
|
if (inputCaptured && GetCapture() == getOsWindowHandle()) ReleaseCapture();
|
|
#else
|
|
#warning port me
|
|
#endif
|
|
|
|
for (int i = 0;i < ghosthwnd.getNumItems();i++)
|
|
Wasabi::Std::Wnd::destroyWnd(ghosthwnd.enumItem(i));
|
|
|
|
if (hwnd != NULL && !m_takenOver)
|
|
{
|
|
#ifdef URLDROPS
|
|
if (acceptExternalDrops()) Wasabi::Std::Wnd::revokeDragNDrop(hwnd /*, &m_target*/);
|
|
#else
|
|
#ifndef WA3COMPATIBILITY
|
|
if (m_target != NULL)
|
|
{
|
|
Wasabi::Std::Wnd::revokeDragNDrop(hwnd);
|
|
}
|
|
#endif
|
|
#endif
|
|
int popact = !wantActivation();
|
|
if (popact) WASABI_API_WND->appdeactivation_push_disallow(this);
|
|
|
|
Wasabi::Std::Wnd::destroyWnd(hwnd);
|
|
|
|
if (popact) WASABI_API_WND->appdeactivation_pop_disallow(this);
|
|
}
|
|
|
|
deleteFrameBuffer(virtualCanvas);
|
|
virtualCanvas = NULL;
|
|
delete scalecanvas;
|
|
scalecanvas = NULL;
|
|
|
|
resetDragSet();
|
|
|
|
notifyParent(ChildNotify::DELETED);
|
|
if (tipdestroytimer)
|
|
killTimer(TIP_DESTROYTIMER_ID);
|
|
if (tipshowtimer)
|
|
{
|
|
// TODO: on the mac, use CreateMouseTrackingRegion
|
|
TRACKMOUSEEVENT tracker;
|
|
tracker.cbSize=sizeof(tracker);
|
|
tracker.dwFlags = TME_HOVER|TME_CANCEL;
|
|
tracker.hwndTrack = this->getOsWindowHandle();
|
|
tracker.dwHoverTime = TIP_TIMER_THRESHOLD;
|
|
|
|
TrackMouseEvent(&tracker);
|
|
|
|
}
|
|
if (tipawaytimer)
|
|
killTimer(TIP_AWAY_ID);
|
|
|
|
destroyTip();
|
|
|
|
delete tooltip;
|
|
|
|
if (uiwaslocked)
|
|
killTimer(BUFFEREDMSG_TIMER_ID);
|
|
|
|
if (deferedInvalidRgn)
|
|
delete deferedInvalidRgn;
|
|
|
|
delete composedrgn;
|
|
delete subtractorrgn;
|
|
delete deferedCascadeRepaintRgn;
|
|
|
|
if (parentWnd != NULL)
|
|
parentWnd->unregisterRootWndChild(this);
|
|
|
|
if (!m_takenOver && WASABI_API_WND) WASABI_API_WND->unregisterRootWnd(this);
|
|
hwnd = NULL;
|
|
}
|
|
|
|
int BaseWnd::init(ifc_window *parWnd, int nochild)
|
|
{
|
|
if (parWnd == NULL)
|
|
return 0;
|
|
|
|
OSWINDOWHANDLE phwnd = parWnd->getOsWindowHandle();
|
|
ASSERT(phwnd != NULL);
|
|
|
|
parentWnd = parWnd; // set default parent wnd
|
|
int ret = init(parWnd->getOsModuleHandle(), phwnd, nochild);
|
|
|
|
if (!ret)
|
|
parentWnd = NULL; // abort
|
|
|
|
return ret;
|
|
}
|
|
|
|
int BaseWnd::init(OSMODULEHANDLE moduleHandle, OSWINDOWHANDLE parent, int nochild)
|
|
{
|
|
RECT r;
|
|
int w, h;
|
|
|
|
ASSERTPR(getOsWindowHandle() == NULL, "don't you double init you gaybag");
|
|
|
|
hinstance = moduleHandle;
|
|
|
|
#ifdef _WIN32
|
|
ASSERT(hinstance != NULL);
|
|
#endif
|
|
|
|
//CUT register_wndClass(hinstance);
|
|
|
|
if (parent != NULL)
|
|
{
|
|
Wasabi::Std::Wnd::getClientRect(parent, &r);
|
|
}
|
|
else
|
|
{
|
|
Wasabi::Std::setRect(&r, 0, 0, getPreferences(SUGGESTED_W), getPreferences(SUGGESTED_H));
|
|
}
|
|
|
|
w = (r.right - r.left);
|
|
h = (r.bottom - r.top);
|
|
|
|
rwidth = w;
|
|
rheight = h;
|
|
rx = r.left;
|
|
ry = r.top;
|
|
|
|
int popact = !wantActivation();
|
|
if (popact) WASABI_API_WND->appdeactivation_push_disallow(this);
|
|
|
|
//CUThwnd = createWindow(r.left, r.top, w, h, nochild, parent, hinstance);
|
|
hwnd = Wasabi::Std::Wnd::createWnd(&r, nochild, acceptExternalDrops(), parent, hinstance, static_cast<ifc_window*>(this));
|
|
#ifdef __APPLE__
|
|
#warning remove me
|
|
Wasabi::Std::Wnd::showWnd(hwnd);
|
|
#endif
|
|
|
|
if (popact) WASABI_API_WND->appdeactivation_pop_disallow(this);
|
|
|
|
//ASSERT(hwnd != NULL); // lets fail nicely, this could happen for some win32 reason, we don't want to fail the whole app for it, so lets just fail the wnd
|
|
if (hwnd == NULL) return 0;
|
|
|
|
if (wantActivation()) bringToFront();
|
|
|
|
//CUT nreal++;
|
|
|
|
//FUCKO
|
|
#ifdef _WIN32 // PORT ME
|
|
#ifdef URLDROPS
|
|
if (acceptExternalDrops()) RegisterDragDrop(hwnd, &m_target);
|
|
#else
|
|
#ifndef WA3COMPATIBILITY
|
|
if (!m_target && WASABI_API_WND != NULL)
|
|
m_target = WASABI_API_WND->getDefaultDropTarget();
|
|
if (m_target != NULL)
|
|
{
|
|
RegisterDragDrop(hwnd, (IDropTarget *)m_target);
|
|
}
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
this_visible = 0;
|
|
|
|
onInit();
|
|
|
|
this_visible = !start_hidden;
|
|
|
|
onPostOnInit();
|
|
|
|
return 1;
|
|
}
|
|
|
|
#ifndef WA3COMPATIBILITY
|
|
void BaseWnd::setDropTarget(void *dt)
|
|
{
|
|
#ifdef _WIN32
|
|
if (isVirtual()) return ;
|
|
if (isInited() && m_target != NULL)
|
|
{
|
|
Wasabi::Std::Wnd::revokeDragNDrop(getOsWindowHandle());
|
|
m_target = NULL;
|
|
}
|
|
m_target = dt;
|
|
if (m_target != NULL && isInited())
|
|
{
|
|
RegisterDragDrop(gethWnd(), (IDropTarget *)m_target);
|
|
}
|
|
#else
|
|
#warning port me
|
|
#endif
|
|
}
|
|
|
|
void *BaseWnd::getDropTarget()
|
|
{
|
|
return m_target;
|
|
}
|
|
#endif
|
|
|
|
int BaseWnd::onInit()
|
|
{
|
|
|
|
const wchar_t *s = getName();
|
|
if (s != NULL)
|
|
setOSWndName(s);
|
|
|
|
inited = 1;
|
|
|
|
if (getParent())
|
|
getParent()->registerRootWndChild(this);
|
|
|
|
if (WASABI_API_WND != NULL)
|
|
WASABI_API_WND->registerRootWnd(this);
|
|
|
|
#ifdef _WIN32
|
|
if (!Wasabi::Std::Wnd::isDesktopAlphaAvailable())
|
|
w2k_alpha = 0; //FUCKO
|
|
|
|
if (w2k_alpha)
|
|
{
|
|
setLayeredWindow(1);
|
|
}
|
|
|
|
if (WM_NULL == WINAMP_WM_DIRECT_MOUSE_WHEEL)
|
|
WINAMP_WM_DIRECT_MOUSE_WHEEL = RegisterWindowMessageW(L"WINAMP_WM_DIRECT_MOUSE_WHEEL");
|
|
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onPostOnInit()
|
|
{
|
|
postoninit = 1; // from now on, isInited() returns 1;
|
|
if (want_autoresize_after_init) onResize();
|
|
else invalidateWindowRegion();
|
|
if (isVisible()) onSetVisible(1);
|
|
if (getTabOrder() == -1) setAutoTabOrder();
|
|
ifc_window *dp = getDesktopParent();
|
|
if ((dp == NULL || dp == this) && WASABI_API_TIMER != NULL)
|
|
postDeferredCallback(DEFERREDCB_FOCUSFIRST, 0, 500);
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::setLayeredWindow(int i)
|
|
{
|
|
if (!Wasabi::Std::Wnd::isValidWnd(getOsWindowHandle())) return ;
|
|
if (!isInited()) return ;
|
|
Wasabi::Std::Wnd::setLayeredWnd(getOsWindowHandle(), i);
|
|
#if 0//CUT
|
|
if (i)
|
|
{
|
|
SetWindowLong(getOsWindowHandle(), GWL_EXSTYLE, GetWindowLong(getOsWindowHandle(), GWL_EXSTYLE) & ~WS_EX_LAYERED);
|
|
SetWindowLong(getOsWindowHandle(), GWL_EXSTYLE, GetWindowLong(getOsWindowHandle(), GWL_EXSTYLE) | WS_EX_LAYERED);
|
|
}
|
|
else
|
|
{
|
|
SetWindowLong(getOsWindowHandle(), GWL_EXSTYLE, GetWindowLong(getOsWindowHandle(), GWL_EXSTYLE) & ~WS_EX_LAYERED);
|
|
}
|
|
#endif
|
|
setTransparency(-1);
|
|
}
|
|
|
|
int BaseWnd::getCursorType(int x, int y)
|
|
{
|
|
if (!customdefaultcursor)
|
|
return BASEWND_CURSOR_POINTER;
|
|
return BASEWND_CURSOR_USERSET;
|
|
}
|
|
|
|
void BaseWnd::onSetName()
|
|
{
|
|
if (isInited() && !isVirtual())
|
|
Wasabi::Std::Wnd::setWndName(getOsWindowHandle(), getNameSafe());
|
|
notifyParent(ChildNotify::NAMECHANGED);
|
|
if (accessible)
|
|
accessible->onSetName(getName());
|
|
}
|
|
|
|
OSWINDOWHANDLE BaseWnd::getOsWindowHandle()
|
|
{
|
|
OSWINDOWHANDLE handle;
|
|
|
|
if ( isVirtual() )
|
|
handle = getParent()->getOsWindowHandle();
|
|
else
|
|
handle = hwnd;
|
|
|
|
|
|
return handle;
|
|
}
|
|
|
|
OSMODULEHANDLE BaseWnd::getOsModuleHandle()
|
|
{
|
|
return hinstance;
|
|
}
|
|
|
|
void BaseWnd::onTip()
|
|
{
|
|
tipshowtimer = FALSE;
|
|
tip_done = TRUE;
|
|
|
|
POINT p;
|
|
Wasabi::Std::getMousePos(&p);
|
|
if (WASABI_API_WND->rootWndFromPoint(&p) == (ifc_window *)this)
|
|
{
|
|
createTip();
|
|
setTimer(TIP_DESTROYTIMER_ID, TIP_LENGTH);
|
|
tipdestroytimer = TRUE;
|
|
}
|
|
setTimer(TIP_AWAY_ID, TIP_AWAY_DELAY);
|
|
tipawaytimer = TRUE;
|
|
}
|
|
|
|
void BaseWnd::timerCallback(int id)
|
|
{
|
|
switch (id)
|
|
{
|
|
case BUFFEREDMSG_TIMER_ID:
|
|
checkLockedUI();
|
|
break;
|
|
// case TIP_TIMER_ID:
|
|
//onTip();
|
|
//break;
|
|
case TIP_DESTROYTIMER_ID:
|
|
killTimer(TIP_DESTROYTIMER_ID);
|
|
killTimer(TIP_AWAY_ID);
|
|
tipawaytimer = FALSE;
|
|
tipdestroytimer = FALSE;
|
|
destroyTip();
|
|
break;
|
|
case TIP_AWAY_ID:
|
|
onTipMouseMove();
|
|
break;
|
|
}
|
|
}
|
|
|
|
int BaseWnd::isInited()
|
|
{
|
|
return inited;
|
|
}
|
|
|
|
int BaseWnd::isDestroying()
|
|
{
|
|
return destroying;
|
|
}
|
|
|
|
int BaseWnd::wantSiblingInvalidations()
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
void BaseWnd::setRSize(int x, int y, int w, int h)
|
|
{
|
|
rwidth = w;
|
|
rheight = h;
|
|
rx = x;
|
|
ry = y;
|
|
}
|
|
|
|
void BaseWnd::resize(int x, int y, int w, int h, int wantcb)
|
|
{
|
|
inonresize = 1;
|
|
|
|
if (x == AUTOWH) x = NOCHANGE;
|
|
if (y == AUTOWH) y = NOCHANGE;
|
|
if (w == AUTOWH) w = NOCHANGE;
|
|
if (h == AUTOWH) h = NOCHANGE;
|
|
|
|
if (getNumMinMaxEnforcers() > 0)
|
|
{
|
|
int min_w = getPreferences(MINIMUM_W);
|
|
int min_h = getPreferences(MINIMUM_H);
|
|
int max_w = getPreferences(MAXIMUM_W);
|
|
int max_h = getPreferences(MAXIMUM_H);
|
|
if (min_w != AUTOWH && w != NOCHANGE && w < min_w) w = min_w;
|
|
if (max_w != AUTOWH && w != NOCHANGE && w > max_w) w = max_w;
|
|
if (min_h != AUTOWH && h != NOCHANGE && h < min_h) h = min_h;
|
|
if (max_h != AUTOWH && h != NOCHANGE && h > max_h) h = max_h;
|
|
}
|
|
|
|
int noresize = (w == NOCHANGE && h == NOCHANGE);
|
|
int nomove = (x == NOCHANGE && y == NOCHANGE)/* || (x == rx && y == ry)*/;
|
|
if (x == NOCHANGE) x = rx;
|
|
if (y == NOCHANGE) y = ry;
|
|
if (w == NOCHANGE) w = rwidth;
|
|
if (h == NOCHANGE) h = rheight;
|
|
|
|
#ifdef _DEBUG
|
|
ASSERT(x < 0xFFF0);
|
|
ASSERT(y < 0xFFF0);
|
|
ASSERT(w < 0xFFF0);
|
|
ASSERT(h < 0xFFF0);
|
|
#endif
|
|
|
|
double thisratio = getRenderRatio();
|
|
int different_ratio = (lastratio != thisratio);
|
|
lastratio = thisratio;
|
|
|
|
int noevent = (resizecount > 1 && w == rwidth && h == rheight);
|
|
//ifc_window *dp = getDesktopParent();
|
|
if (different_ratio == 1 && noevent == 1)
|
|
{
|
|
if (Wasabi::Std::Wnd::getTopmostChild(getOsWindowHandle()) != INVALIDOSWINDOWHANDLE)
|
|
noevent = 0;
|
|
invalidateWindowRegion();
|
|
}
|
|
|
|
RECT oldsize, newsize = Wasabi::Std::makeRect(x, y, w, h);
|
|
if (hwnd != NULL)
|
|
BaseWnd::getNonClientRect(&oldsize);
|
|
else
|
|
oldsize = newsize;
|
|
|
|
setRSize(x, y, w, h);
|
|
|
|
if (handleRatio() && renderRatioActive())
|
|
{
|
|
multRatio(&w, &h);
|
|
if (getParent() != NULL)
|
|
{
|
|
multRatio(&x, &y);
|
|
}
|
|
}
|
|
|
|
if (!noevent)
|
|
{
|
|
if (wantcb && isPostOnInit())
|
|
{
|
|
resizecount = MIN(5, ++resizecount);
|
|
if (!isVirtual())
|
|
invalidateWindowRegion();
|
|
onResize();
|
|
if (ensureWindowRegionValid())
|
|
updateWindowRegion();
|
|
}
|
|
}
|
|
|
|
if (getOsWindowHandle() != NULL)
|
|
{
|
|
|
|
RECT oldsizescaled;
|
|
getWindowRect(&oldsizescaled);
|
|
RECT newsizescaled = {x, y, x + w, y + h};
|
|
if (MEMCMP(&newsizescaled, &oldsizescaled, sizeof(RECT)))
|
|
{
|
|
//CUT SetWindowPos(getOsWindowHandle(), NULL, x, y, w, h,
|
|
//CUT SWP_NOZORDER |
|
|
//CUT SWP_NOACTIVATE |
|
|
//CUT (!wantRedrawOnResize() ? SWP_NOCOPYBITS: 0) |
|
|
//CUT (ncb ? SWP_NOCOPYBITS : 0) |
|
|
//CUT ( nomove ? SWP_NOMOVE : 0) |
|
|
//CUT ( noresize ? SWP_NOSIZE : 0) |
|
|
//CUT 0);
|
|
Wasabi::Std::Wnd::setWndPos( getOsWindowHandle(), NULL, x, y, w, h, TRUE, TRUE, !wantRedrawOnResize() || ncb, nomove, noresize );
|
|
}
|
|
//else
|
|
//{
|
|
// DebugStringW(L"BaseWnd::resize optimized\n");
|
|
//}
|
|
|
|
onAfterResize();
|
|
|
|
if (ncb)
|
|
invalidate();
|
|
else
|
|
{
|
|
RECT r;
|
|
if (hwnd != NULL)
|
|
{
|
|
if (newsize.left == oldsize.left && newsize.top == oldsize.top)
|
|
{
|
|
if (newsize.right > oldsize.right)
|
|
{
|
|
// growing in width
|
|
r.left = oldsize.right;
|
|
r.right = newsize.right;
|
|
r.top = newsize.top;
|
|
r.bottom = newsize.bottom;
|
|
invalidateRect(&r);
|
|
if (newsize.bottom > oldsize.bottom)
|
|
{
|
|
// growing in width & height
|
|
r.left = oldsize.left;
|
|
r.right = newsize.right;
|
|
r.top = oldsize.bottom;
|
|
r.bottom = newsize.bottom;
|
|
invalidateRect(&r);
|
|
}
|
|
}
|
|
else if (newsize.bottom > oldsize.bottom)
|
|
{
|
|
if (newsize.bottom > oldsize.bottom)
|
|
{
|
|
// growing in height
|
|
r.left = oldsize.left;
|
|
r.right = newsize.right;
|
|
r.top = oldsize.bottom;
|
|
r.bottom = newsize.bottom;
|
|
invalidateRect(&r);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
inonresize = 0;
|
|
}
|
|
|
|
void BaseWnd::forcedOnResizeChain(ifc_window *w)
|
|
{
|
|
w->triggerEvent(TRIGGER_ONRESIZE);
|
|
int n = w->getNumRootWndChildren();
|
|
for (int i = 0;i < n;i++)
|
|
{
|
|
forcedOnResizeChain(w->enumRootWndChildren(i));
|
|
}
|
|
}
|
|
|
|
int BaseWnd::forcedOnResize()
|
|
{
|
|
forcedOnResizeChain(this);
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::onResize()
|
|
{
|
|
if (!isVirtual() || (getRegionOp() != REGIONOP_NONE))
|
|
invalidateWindowRegion();
|
|
// you are not supposed to call onResize until after onInit has returned. If what you wanted was to generate
|
|
// an onResize event to do some custom client coordinates recalculations (ie: to apply on your children)
|
|
// then you don't need to do anything since onResize is going to be called after onInit() is done. If you still want to
|
|
// trigger it because your code might be called by onInit and after onInit, use isPostOnInit() as a test.
|
|
// if what you wanted was to signal a object that you just resized it, then you don't need to do anything beside
|
|
// resize(...), it will generate the event on its own if the window is inited, and will defer to until after onInit
|
|
// if it is not.
|
|
// shortly put: do not call onResize before or inside onInit()
|
|
// if you have any valid reason for doing that, i'd like to know about it so i can make it possible. -FG
|
|
#ifdef _DEBUG
|
|
if (!isPostOnInit())
|
|
{
|
|
//__asm int 3;
|
|
ASSERTPR(isPostOnInit(), "do not call onResize before or inside onInit()");
|
|
}
|
|
#endif
|
|
return FALSE;
|
|
}
|
|
|
|
void BaseWnd::resizeToClient(BaseWnd *wnd)
|
|
{
|
|
if (wnd != NULL)
|
|
wnd->resize(&clientRect());
|
|
}
|
|
|
|
int BaseWnd::onPostedMove()
|
|
{
|
|
/*
|
|
if (w2k_alpha && Wasabi::Std::Wnd::isDesktopAlphaAvailable() && !cloaked)
|
|
{
|
|
RECT r;
|
|
getWindowRect(&r);
|
|
Wasabi::Std::Wnd::moveLayeredWnd(hwnd, r.left, r.top);
|
|
}*/
|
|
return FALSE;
|
|
}
|
|
|
|
void BaseWnd::resize(RECT *r, int wantcb)
|
|
{
|
|
resize(r->left, r->top, r->right - r->left, r->bottom - r->top, wantcb);
|
|
}
|
|
|
|
void BaseWnd::move(int x, int y)
|
|
{
|
|
//DebugStringW( L"BaseWnd::move( x = %d, y = %d )\n", x, y );
|
|
|
|
setRSize(x, y, rwidth, rheight);
|
|
Wasabi::Std::Wnd::setWndPos( getOsWindowHandle(), NULL, x, y, 0, 0, TRUE, TRUE, ncb, FALSE, TRUE );
|
|
//CUT if (!ncb)
|
|
//CUT SetWindowPos(getOsWindowHandle(), NULL, x, y, 0, 0, SWP_NOSIZE|SWP_NOZORDER|SWP_NOACTIVATE|SWP_DEFERERASE);
|
|
//CUT else
|
|
//CUT SetWindowPos(getOsWindowHandle(), NULL, x, y, 0, 0, SWP_NOSIZE|SWP_NOZORDER|SWP_NOCOPYBITS|SWP_NOACTIVATE|SWP_DEFERERASE);
|
|
}
|
|
|
|
#ifdef EXPERIMENTAL_INDEPENDENT_AOT
|
|
BOOL CALLBACK EnumOwnedTopMostWindows(HWND hwnd, LPARAM lParam)
|
|
{
|
|
enumownedstruct *st = (enumownedstruct *)lParam;
|
|
if (hwnd != st->hthis && GetWindow(hwnd, GW_OWNER) == st->owner)
|
|
{
|
|
ifc_window *w = (ifc_window*)GetWindowLong(hwnd, GWL_USERDATA);
|
|
if (w != NULL && w->getAlwaysOnTop())
|
|
st->hlist->addItem(w);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void BaseWnd::saveTopMosts()
|
|
{
|
|
HWND owner = GetWindow(getOsWindowHandle(), GW_OWNER);
|
|
enumownedstruct st;
|
|
ontoplist.removeAll();
|
|
if (owner != NULL)
|
|
{
|
|
st.owner = owner;
|
|
st.hlist = &ontoplist;
|
|
st.hthis = getOsWindowHandle();
|
|
EnumWindows(EnumOwnedTopMostWindows, (long)&st);
|
|
}
|
|
}
|
|
|
|
void BaseWnd::restoreTopMosts()
|
|
{
|
|
HWND owner = GetWindow(getOsWindowHandle(), GW_OWNER);
|
|
if (owner != NULL)
|
|
{
|
|
for (int i = 0;i < ontoplist.getNumItems();i++)
|
|
{
|
|
ontoplist.enumItem(i)->setAlwaysOnTop(1);
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void BaseWnd::bringToFront()
|
|
{
|
|
// when we set a window to the top of the zorder (not topmost), win32 finds the owner and removes any topmost flag its children may
|
|
// have because it assumes we want this window over these, which we definitly don't. so we need to first go thru all the owner's children,
|
|
// make a list of the ones with a topmost flag, set this window on top, and set the topmost flags back. yay
|
|
ASSERT(!isVirtual());
|
|
#ifdef EXPERIMENTAL_INDEPENDENT_AOT
|
|
saveTopMosts();
|
|
#endif
|
|
//CUT SetWindowPos(getOsWindowHandle(), HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE|SWP_DEFERERASE|SWP_NOOWNERZORDER);
|
|
Wasabi::Std::Wnd::bringToFront(getOsWindowHandle());
|
|
#ifdef EXPERIMENTAL_INDEPENDENT_AOT
|
|
restoreTopMosts();
|
|
#endif
|
|
}
|
|
|
|
void BaseWnd::bringToBack()
|
|
{
|
|
ASSERT(!isVirtual());
|
|
#ifdef EXPERIMENTAL_INDEPENDENT_AOT
|
|
saveTopMosts();
|
|
#endif
|
|
//CUT SetWindowPos(getOsWindowHandle(), HWND_BOTTOM, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE|SWP_DEFERERASE|SWP_NOOWNERZORDER);
|
|
Wasabi::Std::Wnd::sendToBack(getOsWindowHandle());
|
|
#ifdef EXPERIMENTAL_INDEPENDENT_AOT
|
|
restoreTopMosts();
|
|
#endif
|
|
}
|
|
|
|
void BaseWnd::setVisible(int show)
|
|
{
|
|
int visible = isVisible(1);
|
|
if (!!visible == !!show) return ;
|
|
invalidate();
|
|
this_visible = !!show;
|
|
/*if (!getParent() || getParent() == WASABI_API_WND->main_getRootWnd() && IsWindow(getOsWindowHandle())) {
|
|
if (!show) {
|
|
setLayeredWindow(0);
|
|
if (setLayeredWindowAttributes)
|
|
setLayeredWindowAttributes(hwnd, RGB(0,0,0), 255, LWA_ALPHA);
|
|
} else {
|
|
setLayeredWindow(w2k_alpha);
|
|
}
|
|
}*/
|
|
if (!getParent() || getParent() == WASABI_API_WND->main_getRootWnd() || getParent()->isVisible())
|
|
{
|
|
onSetVisible(show);
|
|
}
|
|
}
|
|
|
|
void BaseWnd::setCloaked(int cloak)
|
|
{
|
|
if (cloaked == cloak) return ;
|
|
cloaked = cloak;
|
|
if (isVirtual()) return ;
|
|
if (cloaked)
|
|
{
|
|
//CUTif (IsWindowVisible(getOsWindowHandle()))
|
|
//CUT ShowWindow(getOsWindowHandle(), SW_HIDE);
|
|
if (Wasabi::Std::Wnd::isWndVisible(getOsWindowHandle()))
|
|
Wasabi::Std::Wnd::hideWnd(getOsWindowHandle());
|
|
}
|
|
else
|
|
{
|
|
if (isVisible(1))
|
|
//CUTShowWindow(getOsWindowHandle(), SW_NORMAL);
|
|
Wasabi::Std::Wnd::showWnd(getOsWindowHandle());
|
|
}
|
|
}
|
|
|
|
|
|
void BaseWnd::onSetVisible(int show)
|
|
{
|
|
/* for debug purposes - don't delete please
|
|
#include "../../../studio/container.h"
|
|
#include "../../../studio/layout.h"
|
|
if (!show && getGuiObject() && STRCASEEQLSAFE(getGuiObject()->guiobject_getId(), "normal")) {
|
|
Layout *l = (Layout *)getInterface(layoutGuid);
|
|
if (l) {
|
|
if (l->getParentContainer() && STRCASEEQLSAFE(l->getParentContainer()->getId(), "main")) {
|
|
DebugString("Hiding main player\n");
|
|
}
|
|
}
|
|
}*/
|
|
if (!isVirtual())
|
|
if (hwnd != NULL)
|
|
if (!cloaked)
|
|
{
|
|
//CUT // SetWindowPos(getOsWindowHandle(),NULL,0,0,0,0,SWP_NOMOVE|SWP_NOSIZE|SWP_NOACTIVATE|SWP_NOZORDER|SWP_SHOWWINDOW);
|
|
//CUT ShowWindow(getOsWindowHandle(), show ? SW_SHOWNA : SW_HIDE);
|
|
if (show)
|
|
Wasabi::Std::Wnd::showWnd(getOsWindowHandle(), TRUE);
|
|
else
|
|
Wasabi::Std::Wnd::hideWnd(getOsWindowHandle());
|
|
}
|
|
/* if (!show)
|
|
postDeferredCallback(0x7849);
|
|
else {*/
|
|
foreach(rootwndchildren)
|
|
ifc_window *w = rootwndchildren.getfor();
|
|
if (w && w->isVisible(1)) // check internal flag only
|
|
w->onSetVisible(show);
|
|
endfor;
|
|
dependent_sendEvent(BaseWnd::depend_getClassGuid(), Event_SETVISIBLE, show);
|
|
//}
|
|
/* if (getDesktopParent() == this) {
|
|
cascadeRepaint(0);
|
|
}*/
|
|
|
|
/*#ifdef WIN32 // os-specific non virtual child wnd support
|
|
if (!isVirtual()) {
|
|
HWND w = GetWindow(getOsWindowHandle(), GW_CHILD);
|
|
while (w != NULL) {
|
|
api_window *rootwnd = (api_window*)GetWindowLong(w, GWL_USERDATA);
|
|
if (rootwnd && rootwnd != this)
|
|
if (rootwnd->isInited())
|
|
rootwnd->onSetVisible(show);
|
|
w = GetWindow(w, GW_HWNDNEXT);
|
|
}
|
|
}
|
|
#endif*/
|
|
if (!isVirtual())
|
|
{
|
|
if (!show)
|
|
{
|
|
deferedInvalidate();
|
|
delete virtualCanvas;
|
|
virtualCanvas = NULL;
|
|
}
|
|
}
|
|
invalidateWindowRegion();
|
|
}
|
|
|
|
void BaseWnd::setEnabled(int en)
|
|
{
|
|
int enabled = isEnabled(1);
|
|
if (!!enabled == !!en) return ;
|
|
invalidate();
|
|
this_enabled = !!en;
|
|
if (!getParent() || getParent() == WASABI_API_WND->main_getRootWnd() || getParent()->isEnabled())
|
|
{
|
|
onEnable(en);
|
|
}
|
|
}
|
|
|
|
int BaseWnd::isEnabled(int within)
|
|
{
|
|
if (!isVirtual() && !getOsWindowHandle()) return 0;
|
|
if (!this_enabled) return 0;
|
|
|
|
if (within) return this_enabled; // whatever, local
|
|
|
|
if (isVirtual()) // virtual, global
|
|
if (getParent())
|
|
return getParent()->isEnabled();
|
|
else
|
|
return 0;
|
|
|
|
// non virtual, global
|
|
//CUT if (GetWindowLong(getOsWindowHandle(), GWL_STYLE) & WS_POPUP) return this_enabled;
|
|
if (Wasabi::Std::Wnd::isPopup(getOsWindowHandle())) return this_enabled;
|
|
//CUT if (!Wasabi::Std::Wnd::isValidWnd(GetParent(gethWnd()))) return this_enabled;
|
|
if (!Wasabi::Std::Wnd::isValidWnd(Wasabi::Std::Wnd::getParent(getOsWindowHandle()))) return this_enabled;
|
|
if (getParent()) return getParent()->isEnabled(); // not a popup, check its parent or fail
|
|
return this_enabled;
|
|
}
|
|
|
|
int BaseWnd::onEnable(int en)
|
|
{
|
|
if (!isVirtual())
|
|
{
|
|
if (hwnd != NULL)
|
|
//CUT EnableWindow(getOsWindowHandle(), en);
|
|
Wasabi::Std::Wnd::setEnabled(getOsWindowHandle(), en);
|
|
foreach(rootwndchildren)
|
|
ifc_window *w = rootwndchildren.getfor();
|
|
if (w->isEnabled(1)) // check internal flag only
|
|
w->onEnable(en);
|
|
endfor;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::setFocus()
|
|
{
|
|
if (curVirtualChildFocus != NULL)
|
|
{
|
|
curVirtualChildFocus->onKillFocus();
|
|
curVirtualChildFocus = NULL;
|
|
}
|
|
onSetRootFocus(this);
|
|
//CUT SetFocus(getOsWindowHandle());
|
|
Wasabi::Std::Wnd::setFocus(getOsWindowHandle());
|
|
}
|
|
|
|
void BaseWnd::setFocusOnClick(int f)
|
|
{
|
|
focus_on_click = f;
|
|
}
|
|
|
|
api_region *BaseWnd::getDeferedInvalidRgn()
|
|
{
|
|
return deferedInvalidRgn;
|
|
}
|
|
|
|
void BaseWnd::deferedInvalidate()
|
|
{
|
|
if (!hasVirtualChildren() || !isVisible(1)) return ;
|
|
RECT r = Wasabi::Std::makeRect(0, 0, 0, 0);
|
|
getNonClientRect(&r);
|
|
deferedInvalidateRect(&r);
|
|
}
|
|
|
|
void BaseWnd::deferedInvalidateRect(RECT *r)
|
|
{
|
|
if (!hasVirtualChildren()) return ;
|
|
RegionI h(r);
|
|
deferedInvalidateRgn(&h);
|
|
}
|
|
|
|
void BaseWnd::deferedInvalidateRgn(api_region *h)
|
|
{
|
|
if (!hasVirtualChildren()) return ;
|
|
if (!deferedInvalidRgn)
|
|
{
|
|
deferedInvalidRgn = new RegionI();
|
|
}
|
|
|
|
deferedInvalidRgn->addRegion(h);
|
|
}
|
|
|
|
void BaseWnd::deferedValidate()
|
|
{
|
|
if (!hasVirtualChildren() || !isVisible(1)) return ;
|
|
RECT r = Wasabi::Std::makeRect(0,0,0,0);
|
|
getNonClientRect(&r);
|
|
deferedValidateRect(&r);
|
|
}
|
|
|
|
void BaseWnd::deferedValidateRect(RECT *r)
|
|
{
|
|
if (!hasVirtualChildren()) return ;
|
|
RegionI h(r);
|
|
deferedValidateRgn(&h);
|
|
}
|
|
|
|
void BaseWnd::deferedValidateRgn(api_region *h)
|
|
{
|
|
if (!hasVirtualChildren()) return ;
|
|
if (!deferedInvalidRgn) return ;
|
|
|
|
deferedInvalidRgn->subtractRgn(h);
|
|
}
|
|
|
|
int BaseWnd::hasVirtualChildren()
|
|
{
|
|
return 1; //virtualChildren.getNumItems() > 0;
|
|
}
|
|
|
|
void BaseWnd::invalidate()
|
|
{
|
|
invalidateFrom(this);
|
|
}
|
|
|
|
void BaseWnd::invalidateFrom(ifc_window *who)
|
|
{
|
|
if (hasVirtualChildren()) deferedInvalidate();
|
|
//CUT if (hwnd != NULL && isVisible(1)) InvalidateRect(getOsWindowHandle(), NULL, FALSE);
|
|
if (hwnd != NULL && isVisible(1))
|
|
Wasabi::Std::Wnd::invalidateRect(getOsWindowHandle());
|
|
}
|
|
|
|
void BaseWnd::invalidateRectFrom(RECT *r, ifc_window *who)
|
|
{
|
|
if (hasVirtualChildren()) deferedInvalidateRect(r);
|
|
RegionI rg(r);
|
|
invalidateRgnFrom(&rg, who);
|
|
}
|
|
|
|
void BaseWnd::invalidateRgn(api_region *r)
|
|
{
|
|
invalidateRgnFrom(r, this);
|
|
}
|
|
|
|
void BaseWnd::invalidateRect(RECT *r)
|
|
{
|
|
invalidateRectFrom(r, this);
|
|
}
|
|
|
|
void BaseWnd::invalidateRgnFrom(api_region *r, ifc_window *who)
|
|
{
|
|
if (parentWnd) parentWnd->onChildInvalidate(r, who);
|
|
PaintCallbackInfoI pc(NULL, r);
|
|
dependent_sendEvent(BaseWnd::depend_getClassGuid(), Event_ONINVALIDATE, 0, &pc);
|
|
if (hwnd != NULL && isVisible(1))
|
|
{
|
|
if (hasVirtualChildren())
|
|
{
|
|
api_region *_r = r->clone();
|
|
int j = virtualChildren.searchItem(who);
|
|
for (int i = 0;i < virtualChildren.getNumItems();i++)
|
|
{
|
|
ifc_window *w = virtualChildren[i];
|
|
if (w != who && w->wantSiblingInvalidations())
|
|
w->onSiblingInvalidateRgn(_r, who, j, i);
|
|
}
|
|
|
|
deferedInvalidateRgn(_r);
|
|
physicalInvalidateRgn(_r);
|
|
r->disposeClone(_r);
|
|
}
|
|
else
|
|
{
|
|
deferedInvalidateRgn(r);
|
|
physicalInvalidateRgn(r);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BaseWnd::physicalInvalidateRgn(api_region *r)
|
|
{
|
|
if (hwnd != NULL && isVisible(1))
|
|
{
|
|
if (renderRatioActive())
|
|
{
|
|
api_region *clone = r->clone();
|
|
clone->scale(getRenderRatio(), getRenderRatio(), TRUE);
|
|
//CUT InvalidateRgn(getOsWindowHandle(), clone->getOSHandle(), FALSE);
|
|
Wasabi::Std::Wnd::invalidateRegion(getOsWindowHandle(), clone->getOSHandle());
|
|
r->disposeClone(clone);
|
|
}
|
|
else
|
|
//CUT InvalidateRgn(getOsWindowHandle(), r->getOSHandle(), FALSE);
|
|
Wasabi::Std::Wnd::invalidateRegion(getOsWindowHandle(), r->getOSHandle());
|
|
}
|
|
}
|
|
|
|
void BaseWnd::validate()
|
|
{
|
|
//CUT if (hwnd != NULL) ValidateRect(getOsWindowHandle(), NULL);
|
|
if (hwnd != NULL)
|
|
Wasabi::Std::Wnd::validateRect(getOsWindowHandle());
|
|
}
|
|
|
|
void BaseWnd::validateRect(RECT *r)
|
|
{
|
|
if (hwnd != NULL)
|
|
{
|
|
if (renderRatioActive())
|
|
{
|
|
RECT r2 = *r;
|
|
Wasabi::Std::scaleRect(&r2, getRenderRatio());
|
|
//CUT ValidateRect(getOsWindowHandle(), &r2);
|
|
Wasabi::Std::Wnd::validateRect(getOsWindowHandle(), &r2);
|
|
}
|
|
else
|
|
//CUT ValidateRect(getOsWindowHandle(), r);
|
|
Wasabi::Std::Wnd::validateRect(getOsWindowHandle(), r);
|
|
}
|
|
}
|
|
|
|
void BaseWnd::validateRgn(api_region *reg)
|
|
{
|
|
if (hwnd != NULL)
|
|
{
|
|
if (renderRatioActive())
|
|
{
|
|
api_region *clone = reg->clone();
|
|
clone->scale(getRenderRatio(), getRenderRatio(), TRUE);
|
|
//CUT ValidateRgn(getOsWindowHandle(), clone->getOSHandle());
|
|
Wasabi::Std::Wnd::validateRegion(getOsWindowHandle(), clone->getOSHandle());
|
|
reg->disposeClone(clone);
|
|
}
|
|
else
|
|
//CUT ValidateRgn(getOsWindowHandle(), reg->getOSHandle());
|
|
Wasabi::Std::Wnd::validateRegion(getOsWindowHandle(), reg->getOSHandle());
|
|
}
|
|
}
|
|
|
|
void BaseWnd::repaint()
|
|
{
|
|
/* if (hasVirtualChildren()) {
|
|
api_region *h = new api_region();
|
|
int s = GetUpdateRgn(getOsWindowHandle(), h->getHRGN(), FALSE);
|
|
if (s != NULLREGION && s != ERROR) {
|
|
virtualDrawRgn(h);
|
|
}
|
|
delete h;
|
|
}*/
|
|
//CUTif (hwnd != NULL) UpdateWindow(getOsWindowHandle());
|
|
if (hwnd != NULL)
|
|
Wasabi::Std::Wnd::update(getOsWindowHandle());
|
|
}
|
|
|
|
void BaseWnd::getClientRect(RECT *rect)
|
|
{
|
|
/* rect->left = rx;
|
|
rect->right = rx + rwidth;
|
|
rect->top = ry;
|
|
rect->bottom = ry + rheight;*/
|
|
//ASSERT(hwnd != NULL);
|
|
if (!Wasabi::Std::Wnd::isValidWnd(hwnd))
|
|
{
|
|
MEMSET(rect, 0, sizeof(RECT));
|
|
return ;
|
|
}
|
|
|
|
GetClientRect(getOsWindowHandle(), rect);
|
|
////Wasabi::Std::Wnd::getClientRect(getOsWindowHandle(), rect);
|
|
rect->right = rect->left + rwidth;
|
|
rect->bottom = rect->top + rheight;
|
|
}
|
|
|
|
RECT BaseWnd::clientRect()
|
|
{
|
|
RECT ret;
|
|
getClientRect(&ret);
|
|
return ret;
|
|
}
|
|
|
|
void BaseWnd::getNonClientRect(RECT *rect)
|
|
{
|
|
// ASSERT(hwnd != NULL);
|
|
if (!hwnd)
|
|
getClientRect(rect);
|
|
else
|
|
{
|
|
Wasabi::Std::Wnd::getClientRect(getOsWindowHandle(), rect);
|
|
if (getRenderRatio() != 1.0)
|
|
{
|
|
rect->right = rect->left + rwidth;
|
|
rect->bottom = rect->left + rheight;
|
|
}
|
|
}
|
|
/* rect->left = rx;
|
|
rect->right = rx + rwidth;
|
|
rect->top = ry;
|
|
rect->bottom = ry + rheight;*/
|
|
}
|
|
|
|
RECT BaseWnd::nonClientRect()
|
|
{
|
|
RECT ret;
|
|
getNonClientRect(&ret);
|
|
return ret;
|
|
}
|
|
|
|
void BaseWnd::getWindowRect(RECT *rect)
|
|
{
|
|
//CUT#ifdef WIN32
|
|
//CUT ASSERT(hwnd != NULL);
|
|
//CUT GetWindowRect(getOsWindowHandle(), rect);
|
|
//CUT#else
|
|
//CUT#error port me
|
|
//CUT#endif
|
|
Wasabi::Std::Wnd::getWindowRect(getOsWindowHandle(), rect);
|
|
}
|
|
|
|
// get position relative to parent (same coordinate system for basewnd & virtualwnd)
|
|
void BaseWnd::getPosition(POINT *pt)
|
|
{
|
|
pt->x = rx;
|
|
pt->y = ry;
|
|
}
|
|
|
|
void *BaseWnd::dependent_getInterface(const GUID *classguid)
|
|
{
|
|
HANDLEGETINTERFACE(ifc_window);
|
|
//CUT HANDLEGETINTERFACE(api_window);
|
|
return NULL;
|
|
}
|
|
|
|
RECT BaseWnd::windowRect()
|
|
{
|
|
RECT ret;
|
|
getWindowRect(&ret);
|
|
return ret;
|
|
}
|
|
|
|
|
|
void BaseWnd::clientToScreen(int *x, int *y)
|
|
{
|
|
int _x = x ? *x : 0;
|
|
int _y = y ? *y : 0;
|
|
if (renderRatioActive())
|
|
{
|
|
_x = (int)((double)_x * getRenderRatio());
|
|
_y = (int)((double)_y * getRenderRatio());
|
|
}
|
|
Wasabi::Std::Wnd::clientToScreen(getOsWindowHandle(), &_x, &_y);
|
|
if (x) *x = _x;
|
|
if (y) *y = _y;
|
|
}
|
|
|
|
void BaseWnd::clientToScreen(RECT *r)
|
|
{
|
|
clientToScreen((int*)&r->left, (int*)&r->top);
|
|
clientToScreen((int*)&r->right, (int*)&r->bottom);
|
|
}
|
|
|
|
void BaseWnd::clientToScreen(POINT *p)
|
|
{
|
|
clientToScreen((int *)&p->x, (int *)&p->y);
|
|
}
|
|
|
|
void BaseWnd::screenToClient(int *x, int *y)
|
|
{
|
|
//CUT POINT p;
|
|
int _x = x ? *x : 0;
|
|
int _y = y ? *y : 0;
|
|
//CUT ScreenToClient(getOsWindowHandle(), &p);
|
|
Wasabi::Std::Wnd::screenToClient(getOsWindowHandle(), &_x, &_y);
|
|
if (renderRatioActive())
|
|
{
|
|
_x = (int)((double)_x / getRenderRatio());
|
|
_y = (int)((double)_y / getRenderRatio());
|
|
}
|
|
if (x) *x = _x;
|
|
if (y) *y = _y;
|
|
}
|
|
|
|
void BaseWnd::screenToClient(RECT *r)
|
|
{
|
|
screenToClient((int*)&r->left, (int*)&r->top);
|
|
screenToClient((int*)&r->right, (int*)&r->bottom);
|
|
}
|
|
|
|
void BaseWnd::screenToClient(POINT *p)
|
|
{
|
|
screenToClient((int *)&p->x, (int *)&p->y);
|
|
}
|
|
|
|
void BaseWnd::setParent(ifc_window *newparent)
|
|
{
|
|
ASSERTPR(newparent != NULL, "quit being a weeny");
|
|
ASSERTPR(parentWnd == NULL || newparent == parentWnd, "can't reset parent");
|
|
parentWnd = newparent;
|
|
if (isInited())
|
|
{
|
|
OSWINDOWHANDLE w1 = getOsWindowHandle();
|
|
OSWINDOWHANDLE w2 = newparent->getOsWindowHandle();
|
|
if (w1 != w2)
|
|
//CUT SetParent(w1, w2);
|
|
Wasabi::Std::Wnd::setParent(w1, w2);
|
|
}
|
|
}
|
|
|
|
//FUCKO
|
|
int BaseWnd::reparent(ifc_window *newparent)
|
|
{
|
|
#ifdef _WIN32
|
|
if (!isVirtual())
|
|
{
|
|
if (isInited())
|
|
{
|
|
ifc_window *old = getParent();
|
|
if (!old && newparent)
|
|
{
|
|
::SetParent(getOsWindowHandle(), newparent->getOsWindowHandle());
|
|
SetWindowLong(getOsWindowHandle() , GWL_STYLE, GetWindowLong(getOsWindowHandle(), GWL_STYLE) & ~WS_POPUP);
|
|
SetWindowLong(getOsWindowHandle() , GWL_STYLE, GetWindowLong(getOsWindowHandle(), GWL_STYLE) | WS_CHILD);
|
|
}
|
|
else if (old && !newparent)
|
|
{
|
|
SetWindowLong(getOsWindowHandle() , GWL_STYLE, GetWindowLong(getOsWindowHandle(), GWL_STYLE) & ~WS_CHILD);
|
|
SetWindowLong(getOsWindowHandle() , GWL_STYLE, GetWindowLong(getOsWindowHandle(), GWL_STYLE) | WS_POPUP);
|
|
::SetParent(getOsWindowHandle(), NULL);
|
|
}
|
|
else
|
|
{
|
|
::SetParent(getOsWindowHandle(), newparent ? newparent->getOsWindowHandle() : NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
parentWnd = newparent;
|
|
onSetParent(newparent);
|
|
|
|
#ifdef WASABI_ON_REPARENT
|
|
WASABI_ON_REPARENT(getOsWindowHandle());
|
|
#endif
|
|
#else
|
|
#warning port me
|
|
#endif
|
|
return 1;
|
|
}
|
|
|
|
ifc_window *BaseWnd::getParent()
|
|
{
|
|
return parentWnd;
|
|
}
|
|
|
|
ifc_window *BaseWnd::getRootParent()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
//PORTME
|
|
ifc_window *BaseWnd::getDesktopParent()
|
|
{
|
|
#ifdef _WIN32
|
|
// NONPORTABLE
|
|
HWND w = getOsWindowHandle();
|
|
HWND last = w;
|
|
if (!w) return NULL;
|
|
HWND p = w;
|
|
wchar_t cn[256] = {0};
|
|
while (p && !(GetWindowLong(p, GWL_STYLE) & WS_POPUP))
|
|
{
|
|
GetClassNameW(p, cn, 255); cn[255] = 0;
|
|
if (!wcscmp(cn, BASEWNDCLASSNAME))
|
|
last = p;
|
|
p = GetParent(p);
|
|
}
|
|
if (p)
|
|
{
|
|
GetClassNameW(p, cn, 255); cn[255] = 0;
|
|
if (!wcscmp(cn, BASEWNDCLASSNAME))
|
|
return (ifc_window*)GetWindowLongPtrW(p, GWLP_USERDATA);
|
|
else if (last != NULL)
|
|
return (ifc_window*)GetWindowLongPtrW(last, GWLP_USERDATA);
|
|
}
|
|
#else
|
|
#warning port me
|
|
#endif
|
|
return NULL;
|
|
}
|
|
|
|
int BaseWnd::notifyParent(int msg, int param1, int param2)
|
|
{
|
|
ifc_window *notifywnd = getNotifyWindow();
|
|
if (getParent() == NULL && notifywnd == NULL) return 0;
|
|
if (notifywnd == NULL) notifywnd = getParent();
|
|
ASSERT(notifywnd != NULL);
|
|
return notifywnd->childNotify(this, msg, param1, param2);
|
|
}
|
|
|
|
int BaseWnd::passNotifyUp(ifc_window *child, int msg, int param1, int param2)
|
|
{
|
|
// Same code as above to decide for whom we should notify.
|
|
ifc_window *notifywnd = getNotifyWindow();
|
|
if (getParent() == NULL && notifywnd == NULL) return 0;
|
|
if (notifywnd == NULL) notifywnd = getParent();
|
|
ASSERT(notifywnd != NULL);
|
|
// And here we just change the api_window pointer.
|
|
return notifywnd->childNotify(child, msg, param1, param2);
|
|
}
|
|
|
|
void BaseWnd::setNotifyId(int id)
|
|
{
|
|
notifyid = id;
|
|
}
|
|
|
|
int BaseWnd::getNotifyId()
|
|
{
|
|
return notifyid;
|
|
}
|
|
|
|
DragInterface *BaseWnd::getDragInterface()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
ifc_window *BaseWnd::rootWndFromPoint(POINT *pt)
|
|
{
|
|
// pt is in client coordinates
|
|
int x = (int)((double)pt->x / getRenderRatio());
|
|
int y = (int)((double)pt->y / getRenderRatio());
|
|
|
|
ifc_window *ret = findRootWndChild(x, y);
|
|
if (ret == NULL) ret = this;
|
|
return ret;
|
|
}
|
|
|
|
int BaseWnd::rootwnd_paintTree(ifc_canvas *canvas, api_region *r)
|
|
{
|
|
BaseCloneCanvas c(canvas);
|
|
return paintTree(&c, r);
|
|
}
|
|
|
|
const wchar_t *BaseWnd::getRootWndName()
|
|
{
|
|
return getName();
|
|
}
|
|
|
|
const wchar_t *BaseWnd::getId()
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
void BaseWnd::setSkinId(int id)
|
|
{
|
|
skin_id = id;
|
|
}
|
|
|
|
void BaseWnd::setPreferences(int what, int v)
|
|
{
|
|
switch (what)
|
|
{
|
|
case MAXIMUM_W: maximum_w = v; break;
|
|
case MAXIMUM_H: maximum_h = v; break;
|
|
case MINIMUM_W: minimum_w = v; break;
|
|
case MINIMUM_H: minimum_h = v; break;
|
|
case SUGGESTED_W: suggested_w = v; break;
|
|
case SUGGESTED_H: suggested_h = v; break;
|
|
}
|
|
}
|
|
|
|
int BaseWnd::getPreferences(int what)
|
|
{
|
|
if (getNumMinMaxEnforcers() > 0)
|
|
{
|
|
|
|
int min_x = minimum_w, min_y = minimum_h, max_x = maximum_w, max_y = maximum_h, sug_x = suggested_w, sug_y = suggested_h;
|
|
|
|
for (int i = 0;i < getNumMinMaxEnforcers();i++)
|
|
{
|
|
|
|
int tmin_x = MINIMUM_W, tmin_y = MINIMUM_H, tmax_x = MAXIMUM_W, tmax_y = MAXIMUM_H, tsug_x = SUGGESTED_W, tsug_y = SUGGESTED_H;
|
|
|
|
ifc_window *w = enumMinMaxEnforcer(i);
|
|
|
|
if (w)
|
|
{
|
|
|
|
tmin_x = w->getPreferences(MINIMUM_W);
|
|
tmin_y = w->getPreferences(MINIMUM_H);
|
|
tmax_x = w->getPreferences(MAXIMUM_W);
|
|
tmax_y = w->getPreferences(MAXIMUM_H);
|
|
tsug_x = w->getPreferences(SUGGESTED_W);
|
|
tsug_y = w->getPreferences(SUGGESTED_H);
|
|
|
|
if (tmin_x == -1) tmin_x = AUTOWH;
|
|
if (tmin_y == -1) tmin_y = AUTOWH;
|
|
if (tmax_x == -1) tmax_x = AUTOWH;
|
|
if (tmax_y == -1) tmax_y = AUTOWH;
|
|
if (tsug_x == -1) tsug_x = AUTOWH;
|
|
if (tsug_y == -1) tsug_y = AUTOWH;
|
|
|
|
#ifndef DISABLE_SYSFONTSCALE
|
|
TextInfoCanvas textInfoCanvas(this);
|
|
double fontScale = textInfoCanvas.getSystemFontScale();
|
|
GuiObject *o = static_cast<GuiObject *>(getInterface(guiObjectGuid));
|
|
if (o != NULL)
|
|
{
|
|
if (o->guiobject_getAutoSysMetricsW())
|
|
{
|
|
if (tmin_x != AUTOWH) tmin_x = (int)((float)tmin_x * fontScale);
|
|
if (tmax_x != AUTOWH) tmax_x = (int)((float)tmax_x * fontScale);
|
|
if (tsug_x != AUTOWH) tsug_x = (int)((float)tsug_x * fontScale);
|
|
}
|
|
if (o->guiobject_getAutoSysMetricsH())
|
|
{
|
|
if (tmin_y != AUTOWH) tmin_y = (int)((float)tmin_y * fontScale);
|
|
if (tmax_y != AUTOWH) tmax_y = (int)((float)tmax_y * fontScale);
|
|
if (tsug_y != AUTOWH) tsug_y = (int)((float)tsug_y * fontScale);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
RECT cor;
|
|
w->getNonClientRect(&cor);
|
|
RECT wr;
|
|
getNonClientRect(&wr);
|
|
|
|
int xdif = (wr.right - wr.left) - (cor.right - cor.left);
|
|
int ydif = (wr.bottom - wr.top) - (cor.bottom - cor.top);
|
|
if (tmin_x != AUTOWH) tmin_x += xdif;
|
|
if (tmin_y != AUTOWH) tmin_y += ydif;
|
|
if (tmax_x != AUTOWH) tmax_x += xdif;
|
|
if (tmax_y != AUTOWH) tmax_y += ydif;
|
|
if (tsug_x != AUTOWH) tsug_x += xdif;
|
|
if (tsug_y != AUTOWH) tsug_y += ydif;
|
|
}
|
|
|
|
if (min_x != AUTOWH) min_x = (tmin_x != AUTOWH) ? MAX(min_x, tmin_x) : min_x; else min_x = tmin_x;
|
|
if (max_x != AUTOWH) max_x = (tmax_x != AUTOWH) ? MAX(max_x, tmax_x) : max_x; else max_x = tmax_x;
|
|
if (min_y != AUTOWH) min_y = (tmin_y != AUTOWH) ? MAX(min_y, tmin_y) : min_y; else min_y = tmin_y;
|
|
if (max_y != AUTOWH) max_y = (tmax_y != AUTOWH) ? MAX(max_y, tmax_y) : max_y; else max_y = tmax_y;
|
|
if (sug_x != AUTOWH) sug_x = (tsug_x != AUTOWH) ? MAX(sug_x, tsug_x) : sug_x; else sug_x = tsug_x;
|
|
if (sug_y != AUTOWH) sug_y = (tsug_y != AUTOWH) ? MAX(sug_y, tsug_y) : sug_y; else sug_y = tsug_y;
|
|
}
|
|
|
|
if (min_x != AUTOWH && min_x == max_x) sug_x = min_x;
|
|
if (min_y != AUTOWH && min_y == max_y) sug_y = min_y;
|
|
|
|
switch (what)
|
|
{
|
|
case MINIMUM_W: return min_x;
|
|
case MINIMUM_H: return min_y;
|
|
case MAXIMUM_W: return max_x;
|
|
case MAXIMUM_H: return max_y;
|
|
case SUGGESTED_W: return sug_x;
|
|
case SUGGESTED_H: return sug_y;
|
|
}
|
|
}
|
|
|
|
switch (what)
|
|
{
|
|
case SUGGESTED_W: return suggested_w;
|
|
case SUGGESTED_H: return suggested_h;
|
|
case MAXIMUM_W: return maximum_w;
|
|
case MAXIMUM_H: return maximum_h;
|
|
case MINIMUM_W: return minimum_w;
|
|
case MINIMUM_H: return minimum_h;
|
|
}
|
|
|
|
return AUTOWH;
|
|
}
|
|
|
|
void BaseWnd::setStartHidden(int wtf)
|
|
{
|
|
start_hidden = wtf;
|
|
}
|
|
|
|
//PORTME
|
|
#ifdef _WIN32
|
|
|
|
|
|
|
|
#define EQUAL_CLSNAME(__name1, __name2)\
|
|
(CSTR_EQUAL == CompareStringW(MAKELCID(MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), SORT_DEFAULT),\
|
|
NORM_IGNORECASE, (__name1), -1, (__name2), -1))
|
|
|
|
static BOOL BaseWnd_IsFrameWindow(HWND hwnd)
|
|
{
|
|
WCHAR szClass[64] = {0};
|
|
if (NULL == hwnd || !GetClassNameW(hwnd, szClass, ARRAYSIZE(szClass)))
|
|
return FALSE;
|
|
|
|
return EQUAL_CLSNAME(szClass, L"Winamp v1.x") ||
|
|
EQUAL_CLSNAME(szClass, L"BaseWindow_RootWnd");
|
|
}
|
|
|
|
|
|
LRESULT BaseWnd::wndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
if (!isDestroying()) switch (uMsg)
|
|
{
|
|
case WM_DEFER_CALLBACK:
|
|
timerclient_onDeferredCallback(wParam, lParam);
|
|
break;
|
|
case WM_SYSCOMMAND:
|
|
{
|
|
if ((wParam & 0xfff0) == SC_SCREENSAVE || (wParam & 0xfff0) == SC_MONITORPOWER)
|
|
{
|
|
ifc_window *main = WASABI_API_WND->main_getRootWnd();
|
|
if (main && main != this)
|
|
return SendMessageW(main->gethWnd(), uMsg, wParam, lParam);
|
|
}
|
|
break;
|
|
}
|
|
//CUT case WM_CREATE:
|
|
//CUT hwnd = hWnd;
|
|
//CUT break;
|
|
|
|
//CUT case WM_CLOSE:
|
|
//CUT return 0;
|
|
|
|
case WM_PAINT:
|
|
{
|
|
if (inonresize && !wantRedrawOnResize()) return 1;
|
|
ASSERT(hwnd != NULL);
|
|
if (!isVisible(1) || IsIconic(hWnd)) break;
|
|
RECT r;
|
|
if (GetUpdateRect(hWnd, &r, FALSE))
|
|
{
|
|
if (virtualOnPaint())
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_PRINTCLIENT:
|
|
{
|
|
bool old_cloaked = (!!cloaked);
|
|
cloaked = true;
|
|
DCCanvas dc((HDC)wParam, this);
|
|
paint(&dc, 0);
|
|
cloaked = old_cloaked;
|
|
|
|
if (lParam & PRF_CHILDREN)
|
|
{
|
|
RECT wnd_size;
|
|
GetWindowRect(hwnd, &wnd_size);
|
|
|
|
HWND child = GetWindow(hwnd, GW_CHILD);
|
|
while (child != NULL)
|
|
{
|
|
if (GetWindowLongPtrW(child, GWL_STYLE) & WS_VISIBLE)
|
|
{
|
|
RECT child_size;
|
|
GetWindowRect(child, &child_size);
|
|
if (child_size.right && child_size.bottom)
|
|
{
|
|
BltCanvas bitmap(child_size.right, child_size.bottom, child);;
|
|
SendMessageW(child, WM_PRINT, (WPARAM)bitmap.getHDC(), PRF_CHILDREN | PRF_CLIENT | PRF_NONCLIENT/*| PRF_OWNED*/);
|
|
//bitmap->makeAlpha(255);
|
|
|
|
//set alpha to 255
|
|
int w, h;
|
|
bitmap.getDim(&w, &h, NULL);
|
|
ARGB32 *m_pBits = (ARGB32 *)bitmap.getBits();
|
|
int nwords = w*h;
|
|
for (; nwords > 0; nwords--, m_pBits++)
|
|
{
|
|
unsigned char *pixel = (unsigned char *)m_pBits;
|
|
pixel[3] = 255; // alpha
|
|
}
|
|
|
|
POINT offset;
|
|
offset.x = child_size.left - wnd_size.left;
|
|
offset.y = child_size.top - wnd_size.top;
|
|
|
|
//BLENDFUNCTION blendFn;
|
|
//blendFn.BlendOp = AC_SRC_OVER;
|
|
//blendFn.BlendFlags = 0;
|
|
//blendFn.SourceConstantAlpha = 255;
|
|
//blendFn.AlphaFormat = 0;
|
|
//AlphaBlend((HDC)wParam, offset.x, offset.y, child_size.right-child_size.left, child_size.bottom-child_size.top,
|
|
// bitmap->getHDC(), 0, 0, child_size.right-child_size.left, child_size.bottom-child_size.top, blendFn);
|
|
StretchBlt((HDC)wParam, offset.x, offset.y, child_size.right-child_size.left, child_size.bottom-child_size.top,
|
|
bitmap.getHDC(), 0, 0, child_size.right-child_size.left, child_size.bottom-child_size.top, SRCCOPY);
|
|
}
|
|
}
|
|
child = GetWindow(child, GW_HWNDNEXT);
|
|
}
|
|
|
|
}
|
|
}
|
|
return 0;
|
|
//CUT case WM_NCPAINT: return 0;
|
|
//CUT case WM_SYNCPAINT: return 0;
|
|
|
|
case WM_SETCURSOR:
|
|
if (checkModal()) return TRUE;
|
|
if (hWnd == (HWND)wParam)
|
|
{
|
|
DWORD windowStyle = (DWORD)GetWindowLongPtrW(hWnd, GWL_STYLE);
|
|
switch(HIWORD(lParam))
|
|
{
|
|
case WM_LBUTTONDOWN:
|
|
case WM_RBUTTONDOWN:
|
|
case WM_MBUTTONDOWN:
|
|
case 0x020B/*WM_XBUTTONDOWN*/:
|
|
DisabledWindow_OnMouseClick(hWnd);
|
|
break;
|
|
}
|
|
int ct = BASEWND_CURSOR_POINTER;
|
|
int _x, _y;
|
|
Wasabi::Std::getMousePos(&_x, &_y);
|
|
screenToClient(&_x, &_y);
|
|
OSCURSORHANDLE c = NULL;
|
|
|
|
if (0 == (WS_DISABLED & windowStyle))
|
|
{
|
|
if (!handleVirtualChildMsg(WM_SETCURSOR, _x, _y, &ct, &c))
|
|
{
|
|
ct = getCursorType(_x, _y);
|
|
}
|
|
}
|
|
wchar_t *wincursor = NULL;
|
|
switch (ct)
|
|
{
|
|
case BASEWND_CURSOR_USERSET:
|
|
|
|
if (c == NULL)
|
|
c = getCustomCursor(_x, _y);
|
|
if (c != NULL)
|
|
{
|
|
SetCursor(c);
|
|
return TRUE;
|
|
}
|
|
else wincursor = IDC_ARROW; // Ensure to have at least a cursor
|
|
break;
|
|
case BASEWND_CURSOR_POINTER:
|
|
wincursor = IDC_ARROW;
|
|
break;
|
|
case BASEWND_CURSOR_NORTHSOUTH:
|
|
wincursor = IDC_SIZENS;
|
|
break;
|
|
case BASEWND_CURSOR_EASTWEST:
|
|
wincursor = IDC_SIZEWE;
|
|
break;
|
|
case BASEWND_CURSOR_NORTHWEST_SOUTHEAST:
|
|
wincursor = IDC_SIZENWSE;
|
|
break;
|
|
case BASEWND_CURSOR_NORTHEAST_SOUTHWEST:
|
|
wincursor = IDC_SIZENESW;
|
|
break;
|
|
case BASEWND_CURSOR_4WAY:
|
|
wincursor = IDC_SIZEALL;
|
|
break;
|
|
case BASEWND_CURSOR_EDIT:
|
|
wincursor = IDC_IBEAM;
|
|
break;
|
|
default:
|
|
wincursor = IDC_ARROW;
|
|
break;
|
|
}
|
|
if (wincursor != NULL)
|
|
{
|
|
SetCursor(LoadCursor(NULL, wincursor));
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
|
|
case WM_TIMER:
|
|
timerCallback((int)wParam);
|
|
return 0;
|
|
|
|
case WM_GETOBJECT:
|
|
if (lParam == OBJID_CLIENT)
|
|
{
|
|
Accessible *acc = getAccessibleObject();
|
|
if (acc != NULL)
|
|
{
|
|
LRESULT lAcc = acc->getOSHandle((int)wParam);
|
|
return lAcc;
|
|
}
|
|
}
|
|
break; // Fall through to DefWindowProc
|
|
|
|
|
|
case WM_SETFOCUS:
|
|
if (!focusEventsEnabled) break;
|
|
if (isInited())
|
|
{
|
|
if (rootfocus != NULL && rootfocus != this)
|
|
{
|
|
if (rootfocus != curVirtualChildFocus)
|
|
rootfocus->setFocus();
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (wantFocus())
|
|
{
|
|
onGetFocus();
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
ifc_window *w = getTab(TAB_GETFIRST);
|
|
if (w != NULL)
|
|
{
|
|
w->setFocus();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_KILLFOCUS:
|
|
{
|
|
ifc_window *rp = getRootParent();
|
|
if (!WASABI_API_WND->rootwndIsValid(rp) || !Wasabi::Std::Wnd::isValidWnd(rp->getOsWindowHandle())) break;
|
|
if (!focusEventsEnabled) break;
|
|
#ifdef WASABI_COMPILE_WND
|
|
if (WASABI_API_WND) WASABI_API_WND->forwardOnKillFocus(); // resets the keyboard active keys buffer
|
|
#endif
|
|
if (!WASABI_API_WND->rootwndIsValid(curVirtualChildFocus)) curVirtualChildFocus = NULL;
|
|
if (curVirtualChildFocus)
|
|
{
|
|
curVirtualChildFocus->onKillFocus();
|
|
curVirtualChildFocus = NULL;
|
|
}
|
|
else
|
|
if (hasfocus) onKillFocus();
|
|
break;
|
|
}
|
|
|
|
// dragging and dropping
|
|
|
|
case WM_LBUTTONDOWN:
|
|
{
|
|
if (lParam == 0xdeadc0de)
|
|
return 1;
|
|
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam))
|
|
return 0;
|
|
|
|
WASABI_API_WND->popupexit_check(this);
|
|
|
|
if (checkModal())
|
|
return 0;
|
|
|
|
abortTip();
|
|
|
|
int xPos = (signed short)LOWORD(lParam);
|
|
int yPos = (signed short)HIWORD(lParam);
|
|
|
|
xPos = (int)((float)xPos / getRenderRatio());
|
|
yPos = (int)((float)yPos / getRenderRatio());
|
|
|
|
if (!getCapture() && hasVirtualChildren() && handleVirtualChildMsg(WM_LBUTTONDOWN, xPos, yPos))
|
|
return 0;
|
|
|
|
if (isEnabled() && !dragging)
|
|
{
|
|
autoFocus(this);
|
|
|
|
int r = 0;
|
|
|
|
if (wantLeftClicks())
|
|
r = onLeftButtonDown(xPos, yPos);
|
|
|
|
if (checkDoubleClick(uMsg, xPos, yPos) && wantDoubleClicks() && onLeftButtonDblClk(xPos, yPos))
|
|
return 0;
|
|
|
|
return r;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_RBUTTONDOWN:
|
|
{
|
|
if (lParam == 0xdeadc0de) return 1;
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
WASABI_API_WND->popupexit_check(this);
|
|
if (checkModal()) return 0;
|
|
abortTip();
|
|
int xPos = (signed short)LOWORD(lParam);
|
|
int yPos = (signed short)HIWORD(lParam);
|
|
xPos = (int)((float)xPos / getRenderRatio());
|
|
yPos = (int)((float)yPos / getRenderRatio());
|
|
if (!getCapture() && hasVirtualChildren())
|
|
if (handleVirtualChildMsg(WM_RBUTTONDOWN, xPos, yPos))
|
|
return 0;
|
|
if (isEnabled() && !dragging)
|
|
{
|
|
autoFocus(this);
|
|
int r = 0;
|
|
if (wantRightClicks())
|
|
r = onRightButtonDown(xPos, yPos);
|
|
if (checkDoubleClick(uMsg, xPos, yPos) && wantDoubleClicks()) if (onRightButtonDblClk(xPos, yPos)) return 0;
|
|
return r;
|
|
}
|
|
}
|
|
break;
|
|
case WM_MOUSEHOVER:
|
|
if (checkModal()) return 0;
|
|
if (!getCapture() && hasVirtualChildren())
|
|
if (handleVirtualChildMsg(WM_MOUSEHOVER, 0, 0))
|
|
return 0;
|
|
break;
|
|
case WM_MOUSEMOVE:
|
|
{
|
|
/* static int mm=0;
|
|
DebugString("mousemove %d\n", mm++);*/
|
|
if (checkModal()) return 0;
|
|
int xPos = (signed short)LOWORD(lParam);
|
|
int yPos = (signed short)HIWORD(lParam);
|
|
xPos = (int)((float)xPos / getRenderRatio());
|
|
yPos = (int)((float)yPos / getRenderRatio());
|
|
if (dragging)
|
|
{
|
|
POINT pt = {xPos, yPos};
|
|
clientToScreen(&pt);
|
|
ifc_window *targ;
|
|
int candrop = 0;
|
|
// find the window the mouse is over
|
|
|
|
targ = NULL;
|
|
if (stickyWnd)
|
|
{
|
|
RECT wr;
|
|
GetWindowRect(stickyWnd->getOsWindowHandle(), &wr);
|
|
if (pt.x >= wr.left - sticky.left &&
|
|
pt.x <= wr.right + sticky.right &&
|
|
pt.y >= wr.top - sticky.top &&
|
|
pt.y <= wr.bottom + sticky.bottom) targ = stickyWnd;
|
|
else stickyWnd = NULL;
|
|
}
|
|
|
|
if (targ == NULL && WASABI_API_WND) targ = WASABI_API_WND->rootWndFromPoint(&pt); // FG> not to self, check
|
|
|
|
DI prevtargdi(prevtarg);
|
|
DI targdi(targ);
|
|
|
|
if (prevtarg != targ)
|
|
{
|
|
// window switch
|
|
if (prevtarg != NULL) prevtargdi.dragLeave(this);
|
|
if (targ != NULL) targdi.dragEnter(this);
|
|
}
|
|
if (targ != NULL)
|
|
candrop = targdi.dragOver(pt.x, pt.y, this);
|
|
if (targ == NULL || !candrop)
|
|
SetCursor(LoadCursor(NULL, MAKEINTRESOURCE(IDC_NO)));
|
|
else
|
|
SetCursor(LoadCursor(NULL, MAKEINTRESOURCE(IDC_APPSTARTING)));
|
|
prevtarg = targ;
|
|
}
|
|
else if (isEnabled())
|
|
{
|
|
tipbeenchecked = FALSE;
|
|
if (!getCapture() && hasVirtualChildren())
|
|
{
|
|
if (handleVirtualChildMsg(WM_MOUSEMOVE, xPos, yPos))
|
|
return 0;
|
|
}
|
|
if (getCapture())
|
|
{
|
|
if (wantMouseMoves())
|
|
if (onMouseMove(xPos, yPos))
|
|
return 0;
|
|
}
|
|
if (!tipbeenchecked) onTipMouseMove();
|
|
return 0;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_LBUTTONUP:
|
|
{
|
|
if (lParam == 0xdeadc0de) return 1;
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (checkModal()) return 0;
|
|
int xPos = (signed short)LOWORD(lParam);
|
|
int yPos = (signed short)HIWORD(lParam);
|
|
xPos = (int)((float)xPos / getRenderRatio());
|
|
yPos = (int)((float)yPos / getRenderRatio());
|
|
abortTip();
|
|
if (!dragging && !getCapture() && hasVirtualChildren())
|
|
{
|
|
if (handleVirtualChildMsg(WM_LBUTTONUP, xPos, yPos))
|
|
return 0;
|
|
}
|
|
if (dragging)
|
|
{
|
|
clientToScreen(&xPos, &yPos);
|
|
int res = 0;
|
|
if (prevtarg != NULL)
|
|
{
|
|
res = DI(prevtarg).dragDrop(this, xPos, yPos);
|
|
}
|
|
|
|
// inform source what happened
|
|
dragComplete(res);
|
|
|
|
resetDragSet();
|
|
prevtarg = NULL;
|
|
stickyWnd = NULL;
|
|
suggestedTitle = NULL;
|
|
SetCursor(LoadCursor(NULL, MAKEINTRESOURCE(IDC_ARROW)));
|
|
Wasabi::Std::Wnd::releaseCapture();
|
|
dragging = 0;
|
|
}
|
|
else if (isEnabled())
|
|
{
|
|
if (wantLeftClicks())
|
|
if (onLeftButtonUp(xPos, yPos)) return 0;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_RBUTTONUP:
|
|
{
|
|
if (lParam == 0xdeadc0de) return 1;
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (checkModal()) return 0;
|
|
abortTip();
|
|
int xPos = (signed short)LOWORD(lParam);
|
|
int yPos = (signed short)HIWORD(lParam);
|
|
xPos = (int)((float)xPos / getRenderRatio());
|
|
yPos = (int)((float)yPos / getRenderRatio());
|
|
if (!getCapture() && hasVirtualChildren())
|
|
{
|
|
if (handleVirtualChildMsg(WM_RBUTTONUP, xPos, yPos))
|
|
return 0;
|
|
}
|
|
if (isEnabled() && !dragging)
|
|
{
|
|
if (wantRightClicks())
|
|
if (onRightButtonUp(xPos, yPos)) return 0;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_CONTEXTMENU:
|
|
{
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (checkModal()) return 0;
|
|
ASSERT(hWnd != NULL);
|
|
int xPos = (signed short)LOWORD(lParam);
|
|
int yPos = (signed short)HIWORD(lParam);
|
|
if (hWnd == getOsWindowHandle())
|
|
{
|
|
if (wantContextMenus())
|
|
if (onContextMenu(xPos, yPos)) return 0;
|
|
}
|
|
else if (GetParent(hWnd) == getOsWindowHandle())
|
|
{
|
|
if (wantContextMenus())
|
|
if (onChildContextMenu(xPos, yPos)) return 0;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_ERASEBKGND:
|
|
return (onEraseBkgnd((HDC)wParam));
|
|
|
|
case WM_MOUSEWHEEL:
|
|
{
|
|
abortTip();
|
|
|
|
int l, a;
|
|
l = (short)HIWORD(wParam) / 120;
|
|
a = (short)HIWORD(wParam);
|
|
if (!l)
|
|
if (a > 0) l = 1;
|
|
else if (a < 0)l = 0;
|
|
a = l >= 0 ? l : -l;
|
|
if (GetAsyncKeyState(VK_MBUTTON)&0x8000)
|
|
{
|
|
if (l >= 0) l = 0; // Fast Forward 5s
|
|
else l = 1; // Rewind 5s
|
|
}
|
|
else
|
|
{
|
|
if (l >= 0) l = 2; // Volume up
|
|
else l = 3; // Volume down
|
|
}
|
|
|
|
int r = 0;
|
|
|
|
if (l & 1)
|
|
r = onMouseWheelDown(!(BOOL)(l & 2), a);
|
|
else
|
|
r = onMouseWheelUp(!(BOOL)(l & 2), a);
|
|
if (r == 0)
|
|
{
|
|
r = WASABI_API_WND->forwardOnMouseWheel(l, a);
|
|
}
|
|
// if it wasn't handled by this wnd, nor by the api, send it to the main wnd, unless we're it
|
|
if (r == 0)
|
|
{
|
|
if (WASABI_API_WND->main_getRootWnd() != this)
|
|
r = (int)SendMessageW(WASABI_API_WND->main_getRootWnd()->gethWnd(), uMsg, wParam, lParam);
|
|
}
|
|
|
|
return r;
|
|
}
|
|
|
|
case WM_WA_RELOAD:
|
|
{
|
|
if (wParam == 0)
|
|
freeResources();
|
|
else
|
|
reloadResources();
|
|
return 0;
|
|
}
|
|
|
|
case WM_WA_GETFBSIZE:
|
|
{
|
|
SIZE *s = (SIZE *)wParam;
|
|
s->cx = rwidth;
|
|
s->cy = rheight;
|
|
return 0;
|
|
}
|
|
|
|
case WM_USER + 8976: // wheel in tip, delete tip
|
|
abortTip();
|
|
return 0;
|
|
|
|
case WM_CHAR:
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (WASABI_API_WND->interceptOnChar((TCHAR) wParam)) return 0;
|
|
if (curVirtualChildFocus == NULL)
|
|
{
|
|
if (onChar(((TCHAR) wParam))) return 0;
|
|
}
|
|
else
|
|
{
|
|
if (curVirtualChildFocus->onChar(((TCHAR) wParam))) return 0;
|
|
}
|
|
if (WASABI_API_WND && WASABI_API_WND->forwardOnChar(this, (TCHAR) wParam, (int)lParam)) return 0;
|
|
break;
|
|
|
|
case WM_KEYDOWN:
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (WASABI_API_WND->interceptOnKeyDown((int) wParam)) return 0;
|
|
if (curVirtualChildFocus == NULL)
|
|
{
|
|
if (onKeyDown((int) wParam)) return 0;
|
|
}
|
|
else
|
|
{
|
|
if (curVirtualChildFocus->onKeyDown((int)wParam)) return 0;
|
|
}
|
|
if (WASABI_API_WND && WASABI_API_WND->forwardOnKeyDown(this, (int) wParam, (int)lParam)) return 0;
|
|
break;
|
|
|
|
case WM_KEYUP:
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (WASABI_API_WND->interceptOnKeyUp((int) wParam)) return 0;
|
|
if (curVirtualChildFocus == NULL)
|
|
{
|
|
if (onKeyUp((int) wParam)) return 0;
|
|
}
|
|
else
|
|
{
|
|
if (curVirtualChildFocus->onKeyUp((int)wParam)) return 0;
|
|
}
|
|
if (WASABI_API_WND && WASABI_API_WND->forwardOnKeyUp(this, (int) wParam, (int)lParam)) return 0;
|
|
break;
|
|
|
|
case WM_SYSKEYDOWN:
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (WASABI_API_WND->interceptOnSysKeyDown((int) wParam, (int)lParam)) return 0;
|
|
if (curVirtualChildFocus == NULL)
|
|
{
|
|
if (onSysKeyDown((int) wParam, (int)lParam)) return 0;
|
|
}
|
|
else
|
|
{
|
|
if (curVirtualChildFocus->onSysKeyDown((int)wParam, (int)lParam)) return 0;
|
|
}
|
|
if (WASABI_API_WND && WASABI_API_WND->forwardOnSysKeyDown(this, (int) wParam, (int)lParam)) return 0;
|
|
break;
|
|
|
|
case WM_SYSKEYUP:
|
|
if (bufferizeLockedUIMsg(uMsg, (int)wParam, (int)lParam)) return 0;
|
|
if (WASABI_API_WND->interceptOnSysKeyUp((int) wParam, (int)lParam)) return 0;
|
|
if (curVirtualChildFocus == NULL)
|
|
{
|
|
if (onSysKeyUp((int) wParam, (int)lParam)) return 0;
|
|
}
|
|
else
|
|
{
|
|
if (curVirtualChildFocus->onSysKeyUp((int)wParam, (int)lParam)) return 0;
|
|
}
|
|
if (WASABI_API_WND && WASABI_API_WND->forwardOnSysKeyUp(this, (int) wParam, (int)lParam)) return 0;
|
|
break;
|
|
|
|
case WM_MOUSEACTIVATE:
|
|
{
|
|
if (checkModal() || !wantActivation())
|
|
return MA_NOACTIVATE;
|
|
//SetFocus(getOsWindowHandle());
|
|
return MA_ACTIVATE;
|
|
}
|
|
|
|
case WM_ACTIVATEAPP:
|
|
|
|
if (wParam == FALSE)
|
|
{
|
|
|
|
if (WASABI_API_WND != NULL)
|
|
{
|
|
WASABI_API_WND->popupexit_signal();
|
|
WASABI_API_SYSCB->syscb_issueCallback(SysCallback::GC, GarbageCollectCallback::GARBAGECOLLECT);
|
|
WASABI_API_WND->kbdReset();
|
|
if (ghosthwnd.getNumItems() > 0 && ghostbust)
|
|
{
|
|
ghostbust = 0; postDeferredCallback(DC_KILLGHOST);
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case WM_ACTIVATE:
|
|
switch(LOWORD(wParam))
|
|
{
|
|
case WA_ACTIVE:
|
|
case WA_CLICKACTIVE:
|
|
if (WASABI_API_WND != NULL)
|
|
WASABI_API_WND->popupexit_check(this);
|
|
|
|
onActivate();
|
|
|
|
if (WA_CLICKACTIVE == LOWORD(wParam))
|
|
{
|
|
POINT pt;
|
|
DWORD pts = GetMessagePos();
|
|
POINTSTOPOINT(pt, pts);
|
|
MapWindowPoints(HWND_DESKTOP, hwnd, &pt, 1);
|
|
HWND hTarget = ChildWindowFromPointEx(hwnd, pt, CWP_SKIPINVISIBLE | CWP_SKIPDISABLED | CWP_SKIPTRANSPARENT);
|
|
if (hTarget && hTarget != hwnd) lastActiveWnd = hTarget;
|
|
}
|
|
|
|
if (lastActiveWnd != hwnd && NULL != lastActiveWnd && IsWindow(lastActiveWnd))
|
|
{
|
|
SendMessageW(lastActiveWnd, uMsg, wParam, lParam);
|
|
return 0;
|
|
}
|
|
break;
|
|
default:
|
|
|
|
onDeactivate();
|
|
|
|
lastActiveWnd = GetFocus();
|
|
|
|
if (NULL != lastActiveWnd && !IsChild(hwnd, lastActiveWnd))
|
|
lastActiveWnd = NULL;
|
|
|
|
{
|
|
#ifndef ARRAYSIZE
|
|
#define ARRAYSIZE(x) (sizeof(x)/sizeof(*x))
|
|
#endif
|
|
if (NULL != lastActiveWnd && !BaseWnd_IsFrameWindow(lastActiveWnd))
|
|
{
|
|
while (lastActiveWnd)
|
|
{
|
|
if (BaseWnd_IsFrameWindow(GetWindow(lastActiveWnd, GW_OWNER)))
|
|
break;
|
|
lastActiveWnd = GetAncestor(lastActiveWnd, GA_PARENT);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (lastActiveWnd != hwnd && NULL != lastActiveWnd)
|
|
{
|
|
SendMessageW(lastActiveWnd, uMsg, wParam, lParam);
|
|
return 0;
|
|
}
|
|
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_NCACTIVATE:
|
|
if (allowDeactivation())
|
|
return TRUE;
|
|
return FALSE;
|
|
|
|
case WM_WINDOWPOSCHANGING:
|
|
{
|
|
if (!isVirtual() && Wasabi::Std::Wnd::isPopup(hwnd))
|
|
{
|
|
WINDOWPOS *wp = (WINDOWPOS *)lParam;
|
|
if (wp->x != rx || wp->y != ry) wp->flags |= SWP_NOMOVE;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case WM_WINDOWPOSCHANGED:
|
|
{
|
|
|
|
WINDOWPOS *lpwp = (WINDOWPOS *)lParam; // points to size and position data
|
|
if (lpwp->flags & SWP_HIDEWINDOW)
|
|
{
|
|
minimized = 1;
|
|
onMinimize();
|
|
}
|
|
else if (lpwp->flags & SWP_SHOWWINDOW)
|
|
{
|
|
minimized = 0;
|
|
onRestore();
|
|
}
|
|
|
|
if (!inonresize)
|
|
{
|
|
int w = rwidth;
|
|
int h = rheight;
|
|
multRatio(&w, &h);
|
|
if (lpwp->cx != w || lpwp->cy != h)
|
|
{
|
|
DebugStringW(L"external onResize\n");
|
|
w = lpwp->cx;
|
|
h = lpwp->cy;
|
|
divRatio(&w, &h);
|
|
setRSize(rx, ry, w, h);
|
|
if (isPostOnInit())
|
|
onResize();
|
|
}
|
|
}
|
|
|
|
onPostedMove();
|
|
return 0;
|
|
}
|
|
|
|
case WM_DROPFILES:
|
|
{
|
|
if (checkModal()) break;
|
|
WASABI_API_WND->pushModalWnd();
|
|
onExternalDropBegin();
|
|
HDROP h = (HDROP)wParam;
|
|
POINT dp = {0};
|
|
DragQueryPoint(h, &dp);
|
|
clientToScreen(&dp);
|
|
// build a file list
|
|
wchar_t buf[WA_MAX_PATH] = {0};
|
|
PtrList<FilenamePS> keep;
|
|
|
|
SetCursor(LoadCursor(NULL, IDC_WAIT));
|
|
|
|
//CUT #if UTF8
|
|
//CUT // doesn't really need UTF8, the "buf" is never written to.
|
|
//CUT // made to be NULL to enforce this concept.
|
|
int nfiles = DragQueryFile(h, 0xffffffff, NULL, 0);
|
|
//CUT #else
|
|
//CUT int nfiles = DragQueryFile(h, 0xffffffff, buf, sizeof(buf));
|
|
//CUT #endif
|
|
|
|
// convert them all to PlayItem *'s
|
|
for (int i = 0; i < nfiles; i++)
|
|
{
|
|
DragQueryFileW(h, i, buf, WA_MAX_PATH);
|
|
addDroppedFile(buf, &keep); // recursive
|
|
|
|
}
|
|
SetCursor(LoadCursor(NULL, IDC_ARROW));
|
|
dragging = 1;
|
|
if (dragEnter(this))
|
|
{
|
|
if (dragOver(dp.x, dp.y, this)) dragDrop(this, dp.x, dp.y);
|
|
}
|
|
else
|
|
{
|
|
dragLeave(this);
|
|
#ifdef FORWARD_DRAGNDROP
|
|
HWND w = WASABI_API_WND->main_getRootWnd()->gethWnd();
|
|
SendMessageW(w, WM_DROPFILES, wParam, lParam);
|
|
#endif
|
|
|
|
}
|
|
dragging = 0;
|
|
|
|
// remove data
|
|
keep.deleteAll();
|
|
resetDragSet();
|
|
|
|
onExternalDropEnd();
|
|
WASABI_API_WND->popModalWnd();
|
|
}
|
|
return 0; // dropfiles
|
|
|
|
case WM_CAPTURECHANGED:
|
|
/* static int cc=0;
|
|
DebugString("capture changed! %d\n", cc++);*/
|
|
if (preventcancelcapture) return 0;
|
|
inputCaptured = 0;
|
|
if (curVirtualChildCaptured != NULL)
|
|
{
|
|
ifc_window *w = curVirtualChildCaptured;
|
|
curVirtualChildCaptured = NULL;
|
|
w->onCancelCapture();
|
|
}
|
|
else
|
|
{
|
|
onCancelCapture();
|
|
}
|
|
return 0;
|
|
|
|
} //switch
|
|
|
|
if (WINAMP_WM_DIRECT_MOUSE_WHEEL == uMsg &&
|
|
WM_NULL != WINAMP_WM_DIRECT_MOUSE_WHEEL)
|
|
{
|
|
wndProc(hWnd, WM_MOUSEWHEEL, wParam, lParam);
|
|
return TRUE;
|
|
}
|
|
|
|
if (uMsg >= WM_USER)
|
|
{
|
|
int ret;
|
|
if (onUserMessage(uMsg, (int)wParam, (int)lParam, &ret))
|
|
return ret;
|
|
return 0;
|
|
}
|
|
|
|
return DefWindowProcW(hWnd, uMsg, wParam, lParam);
|
|
}
|
|
#endif
|
|
int BaseWnd::onUserMessage(int msg, int w, int l, int *r)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::checkDoubleClick(int b, int x, int y)
|
|
{
|
|
#ifdef _WIN32
|
|
uint32_t now = Wasabi::Std::getTickCount();
|
|
|
|
switch (b)
|
|
{
|
|
case WM_LBUTTONDOWN:
|
|
if (lastClick[0] > now - Wasabi::Std::getDoubleClickDelay())
|
|
{
|
|
lastClick[0] = 0;
|
|
if (ABS(lastClickP[0].x - x) > Wasabi::Std::getDoubleClickX() || ABS(lastClickP[0].y - y) > Wasabi::Std::getDoubleClickY()) return 0;
|
|
return 1;
|
|
}
|
|
lastClick[0] = now;
|
|
lastClickP[0].x = x;
|
|
lastClickP[0].y = y;
|
|
break;
|
|
|
|
case WM_RBUTTONDOWN:
|
|
if (lastClick[1] > now - Wasabi::Std::getDoubleClickDelay())
|
|
{
|
|
lastClick[1] = 0;
|
|
if (ABS(lastClickP[1].x - x) > Wasabi::Std::getDoubleClickX() || ABS(lastClickP[1].y - y) > Wasabi::Std::getDoubleClickY()) return 0;
|
|
return 1;
|
|
}
|
|
lastClick[1] = now;
|
|
lastClickP[1].x = x;
|
|
lastClickP[1].y = y;
|
|
break;
|
|
}
|
|
#else
|
|
#warning port me
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onMouseWheelUp(int click, int lines)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onMouseWheelDown(int click, int lines)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onContextMenu(int x, int y)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onChildContextMenu(int x, int y)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onDeferredCallback(intptr_t param1, intptr_t param2)
|
|
{
|
|
switch (param1)
|
|
{
|
|
case DEFERREDCB_FLUSHPAINT:
|
|
do_flushPaint();
|
|
break;
|
|
case DEFERREDCB_INVALIDATE:
|
|
if (isPostOnInit())
|
|
invalidate();
|
|
break;
|
|
case DC_KILLGHOST:
|
|
if (ghosthwnd.getNumItems() > 0)
|
|
{
|
|
preventcancelcapture = 1;
|
|
for (int i = 0;i < ghosthwnd.getNumItems();i++)
|
|
Wasabi::Std::Wnd::destroyWnd(ghosthwnd.enumItem(i));
|
|
ghosthwnd.freeAll();
|
|
preventcancelcapture = 0;
|
|
}
|
|
break;
|
|
case DEFERREDCB_FOCUSFIRST:
|
|
assignRootFocus(NULL);
|
|
if (Wasabi::Std::Wnd::getFocus() == getOsWindowHandle())
|
|
{
|
|
focusNext();
|
|
}
|
|
break;
|
|
case 0x7849 /*DEFERREDCB_ONHIDE*/:
|
|
{
|
|
foreach(rootwndchildren)
|
|
ifc_window *w = rootwndchildren.getfor();
|
|
if (w->isVisible(1)) // check internal flag only
|
|
w->onSetVisible(0);
|
|
endfor;
|
|
dependent_sendEvent(BaseWnd::depend_getClassGuid(), Event_SETVISIBLE, 0);
|
|
break;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onPaint(Canvas *canvas)
|
|
{
|
|
#if 0
|
|
// example:
|
|
PaintCanvas c;
|
|
if (!c.beginPaint(this)) return 0;
|
|
(do some painting)
|
|
c will self -destruct on return
|
|
#endif
|
|
if (renderbasetexture)
|
|
{
|
|
PaintCanvas paintcanvas;
|
|
if (canvas == NULL)
|
|
{
|
|
if (!paintcanvas.beginPaint(this)) return 0;
|
|
canvas = &paintcanvas;
|
|
}
|
|
RECT r;
|
|
getNonClientRect(&r);
|
|
RenderBaseTexture(canvas, r);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onPaint(Canvas *canvas, api_region *h)
|
|
{
|
|
if (!canvas) return onPaint(canvas);
|
|
|
|
#ifdef _WIN32
|
|
int sdc = SaveDC(canvas->getHDC());
|
|
#elif defined(__APPLE__)
|
|
CGContextSaveGState(canvas->getHDC());
|
|
#endif
|
|
|
|
canvas->selectClipRgn(h);
|
|
|
|
int rs = onPaint(canvas);
|
|
|
|
#ifdef _WIN32
|
|
RestoreDC(canvas->getHDC(), sdc);
|
|
#elif defined(__APPLE__)
|
|
CGContextRestoreGState(canvas->getHDC());
|
|
#endif
|
|
|
|
return rs;
|
|
}
|
|
|
|
int BaseWnd::getTransparency()
|
|
{
|
|
return wndalpha;
|
|
}
|
|
|
|
void BaseWnd::setTransparency(int amount)
|
|
{
|
|
//if (wndalpha == amount) return;
|
|
if (amount == 254) amount = 255;
|
|
if (amount == 1) amount = 0;
|
|
|
|
if (amount != -1) wndalpha = amount; else amount = wndalpha;
|
|
|
|
if (!Wasabi::Std::Wnd::isDesktopAlphaAvailable())
|
|
{
|
|
wndalpha = 255;
|
|
return ;
|
|
}
|
|
|
|
if (w2k_alpha)
|
|
{
|
|
invalidate();
|
|
return ;
|
|
}
|
|
|
|
#ifdef WIN32
|
|
|
|
if (!isInited() || isVirtual()) return ;
|
|
if (!Wasabi::Std::Wnd::isValidWnd(getOsWindowHandle())) return ;
|
|
|
|
if (amount < -1) amount = 0;
|
|
else if (amount > 255) amount = 255;
|
|
|
|
//CUT DWORD dwLong = GetWindowLong(hwnd, GWL_EXSTYLE);
|
|
if (amount == 255 && !forceTransparencyFlag())
|
|
{
|
|
Wasabi::Std::Wnd::setLayeredWnd(hwnd, FALSE);
|
|
//CUT if (dwLong & WS_EX_LAYERED)
|
|
//CUT SetWindowLong(hwnd, GWL_EXSTYLE, dwLong & ~WS_EX_LAYERED);
|
|
has_alpha_flag = 0;
|
|
}
|
|
else
|
|
{
|
|
if (!Wasabi::Std::Wnd::isLayeredWnd(hwnd))
|
|
Wasabi::Std::Wnd::setLayeredWnd(hwnd, TRUE);
|
|
//CUT if (!(dwLong & WS_EX_LAYERED))
|
|
//CUT SetWindowLong(hwnd, GWL_EXSTYLE, dwLong | WS_EX_LAYERED);
|
|
Wasabi::Std::Wnd::setLayeredAlpha(hwnd, amount);
|
|
//CUT setLayeredWindowAttributes(hwnd, RGB(0,0,0), amount, LWA_ALPHA);
|
|
has_alpha_flag = 1;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
int BaseWnd::forceTransparencyFlag()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::beginCapture()
|
|
{
|
|
if (!getCapture())
|
|
{
|
|
disable_tooltip_til_recapture = 0;
|
|
curVirtualChildCaptured = NULL;
|
|
/* oldCapture = */Wasabi::Std::Wnd::setCapture(getOsWindowHandle());
|
|
/* if (oldCapture) {
|
|
DebugString("Stolen capture detected, this may be ok, but try to avoid it if possible. Saving old capture\n");
|
|
}*/
|
|
inputCaptured = 1;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::endCapture()
|
|
{
|
|
preventcancelcapture = 1;
|
|
if (Wasabi::Std::Wnd::getCapture() == getOsWindowHandle()) Wasabi::Std::Wnd::releaseCapture();
|
|
/* if (oldCapture) {
|
|
DebugString("Restoring old capture\n");
|
|
SetCapture(oldCapture);
|
|
oldCapture = NULL;
|
|
}*/
|
|
inputCaptured = 0;
|
|
preventcancelcapture = 0;
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::getCapture()
|
|
{
|
|
if (inputCaptured && Wasabi::Std::Wnd::getCapture() == getOsWindowHandle() && curVirtualChildCaptured == NULL) return 1;
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::cancelCapture()
|
|
{
|
|
if (curVirtualChildCaptured != NULL)
|
|
{
|
|
curVirtualChildCaptured->cancelCapture();
|
|
return ;
|
|
}
|
|
if (getCapture()) endCapture();
|
|
onCancelCapture();
|
|
}
|
|
|
|
int BaseWnd::onMouseMove(int x, int y)
|
|
{
|
|
onTipMouseMove();
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::onTipMouseMove()
|
|
{
|
|
POINT p;
|
|
|
|
if (dragging) return ;
|
|
if (disable_tooltip_til_recapture) return ;
|
|
|
|
tipbeenchecked = TRUE;
|
|
|
|
Wasabi::Std::getMousePos(&p);
|
|
|
|
if (WASABI_API_WND->rootWndFromPoint(&p) != (ifc_window *)this)
|
|
{
|
|
// leaving area
|
|
tip_done = FALSE;
|
|
if (tipawaytimer)
|
|
killTimer(TIP_AWAY_ID);
|
|
tipawaytimer = FALSE;
|
|
if (tipshowtimer)
|
|
{
|
|
// TODO: on the mac, use CreateMouseTrackingRegion
|
|
TRACKMOUSEEVENT tracker;
|
|
tracker.cbSize=sizeof(tracker);
|
|
tracker.dwFlags = TME_HOVER|TME_CANCEL;
|
|
tracker.hwndTrack = this->getOsWindowHandle();
|
|
tracker.dwHoverTime = TIP_TIMER_THRESHOLD;
|
|
|
|
TrackMouseEvent(&tracker);
|
|
}
|
|
tipshowtimer = FALSE;
|
|
destroyTip();
|
|
}
|
|
else
|
|
{
|
|
// moving in area
|
|
const wchar_t *t = getTip();
|
|
if (!disable_tooltip_til_recapture && !tipshowtimer && !tip_done && t != NULL && *t != 0)
|
|
{
|
|
//entering area & need tip
|
|
|
|
// TODO: on the mac, use CreateMouseTrackingRegion
|
|
TRACKMOUSEEVENT tracker;
|
|
tracker.cbSize=sizeof(tracker);
|
|
tracker.dwFlags = TME_HOVER;
|
|
tracker.hwndTrack = this->getOsWindowHandle();
|
|
tracker.dwHoverTime = TIP_TIMER_THRESHOLD;
|
|
|
|
TrackMouseEvent(&tracker);
|
|
|
|
tipshowtimer = TRUE;
|
|
}
|
|
/*else if (tipshowtimer)
|
|
{
|
|
TRACKMOUSEEVENT tracker;
|
|
tracker.cbSize=sizeof(tracker);
|
|
tracker.dwFlags = TME_HOVER;
|
|
tracker.hwndTrack = this->getOsWindowHandle();
|
|
tracker.dwHoverTime = TIP_TIMER_THRESHOLD;
|
|
|
|
TrackMouseEvent(&tracker);
|
|
}*/
|
|
}
|
|
}
|
|
|
|
int BaseWnd::onLeftButtonDblClk(int x, int y)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onRightButtonDblClk(int x, int y)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onGetFocus()
|
|
{
|
|
// return TRUE if you override this
|
|
hasfocus = 1;
|
|
notifyParent(ChildNotify::GOTFOCUS);
|
|
getRootParent()->onSetRootFocus(this);
|
|
invalidate();
|
|
Accessible *a = getAccessibleObject();
|
|
if (a != NULL)
|
|
a->onGetFocus();
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::onKillFocus()
|
|
{
|
|
// return TRUE if you override this
|
|
hasfocus = 0;
|
|
notifyParent(ChildNotify::KILLFOCUS);
|
|
invalidate();
|
|
return 1;
|
|
}
|
|
|
|
#if defined(_WIN64)
|
|
int BaseWnd::childNotify(ifc_window* child, int msg, int p1, int p2)
|
|
{
|
|
return 0;
|
|
}
|
|
#else
|
|
int BaseWnd::childNotify(ifc_window *child, int msg, intptr_t p1, intptr_t p2)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
int BaseWnd::addDragItem(const wchar_t *droptype, void *item)
|
|
{
|
|
ASSERT(droptype != NULL);
|
|
if (item == NULL) return -1;
|
|
DragSet *set;
|
|
int pos = dragCheckData(droptype);
|
|
if (pos == -1)
|
|
{
|
|
set = new DragSet();
|
|
set->setName(droptype);
|
|
dragsets.addItem(set);
|
|
pos = dragsets.getNumItems() - 1;
|
|
}
|
|
else set = dragsets[pos];
|
|
set->addItem(item);
|
|
return pos;
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
int BaseWnd::handleDrag()
|
|
{
|
|
abortTip();
|
|
if (dragsets.getNumItems() == 0) return 0;
|
|
|
|
Wasabi::Std::Wnd::setCapture(hwnd);
|
|
SetCursor(LoadCursor(NULL, MAKEINTRESOURCE(IDC_APPSTARTING)));
|
|
|
|
dragging = 1;
|
|
stickyWnd = NULL;
|
|
|
|
return 1;
|
|
}
|
|
#endif
|
|
|
|
int BaseWnd::resetDragSet()
|
|
{
|
|
dragsets.deleteAll();
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::dragEnter(ifc_window *sourceWnd)
|
|
{
|
|
ifc_window *rw = getParent(); //api_window::rootwndFromRootWnd(getParent()); //FG> note to self, check!
|
|
if (rw) return DI(rw).dragEnter(sourceWnd);
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::dragSetSticky(ifc_window *wnd, int left, int right, int up, int down)
|
|
{
|
|
ASSERT(dragging);
|
|
stickyWnd = wnd;
|
|
|
|
if (left < 0) left = 0;
|
|
if (right < 0) right = 0;
|
|
if (up < 0) up = 0;
|
|
if (down < 0) down = 0;
|
|
|
|
Wasabi::Std::setRect(&sticky, left, up, right, down);
|
|
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::setSuggestedDropTitle(const wchar_t *title)
|
|
{
|
|
ASSERT(title != NULL);
|
|
suggestedTitle = title;
|
|
}
|
|
|
|
const wchar_t *BaseWnd::dragGetSuggestedDropTitle()
|
|
{
|
|
return suggestedTitle; // can be NULL
|
|
}
|
|
|
|
int BaseWnd::dragCheckData(const wchar_t *type, int *nitems)
|
|
{
|
|
for (int i = 0; i < dragsets.getNumItems(); i++)
|
|
{
|
|
if (!WCSICMP(type, dragsets[i]->getName()))
|
|
{
|
|
if (nitems != NULL) *nitems = dragsets[i]->getNumItems();
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void *BaseWnd::dragGetData(int slot, int itemnum)
|
|
{
|
|
if (slot < 0 || slot >= dragsets.getNumItems()) return 0;
|
|
if (itemnum < 0 || itemnum >= dragsets[slot]->getNumItems()) return 0;
|
|
return dragsets[slot]->enumItem(itemnum);
|
|
}
|
|
|
|
void BaseWnd::addDroppedFile(const wchar_t *filename, PtrList<FilenamePS> *plist)
|
|
{
|
|
#ifdef RECURSE_SUBDIRS_ON_DROP
|
|
const char *slash = filename + STRLEN(filename) - 1;
|
|
for (; slash > filename; slash--) if (*slash == '/' || *slash == '\\') break;
|
|
if (STREQL(slash + 1, ".") || STREQL(slash + 1, "..")) return ;
|
|
|
|
char buf[WA_MAX_PATH] = {0};
|
|
STRCPY(buf, filename);
|
|
// try to resolve shortcuts
|
|
char *ext = buf + STRLEN(buf) - 1;
|
|
for (; ext > buf; ext--) if (*ext == '.' || *ext == '\\' || *ext == '/') break;
|
|
#ifdef WIN32
|
|
if (!STRICMP(ext, ".lnk"))
|
|
{
|
|
char buf2[MAX_PATH] = {0};
|
|
if (StdFile::resolveShortcut(buf, buf2)) STRCPY(buf, buf2);
|
|
}
|
|
#endif
|
|
|
|
int isdir = 0;
|
|
|
|
// handle root dir specially?
|
|
WIN32_FIND_DATA data = {0};
|
|
HANDLE r = FindFirstFile(buf, &data);
|
|
if (!r) return ;
|
|
FindClose(r);
|
|
if (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) isdir = 1;
|
|
|
|
if (isdir)
|
|
{
|
|
onExternalDropDirScan(buf);
|
|
|
|
// enumerate that dir
|
|
char search[WA_MAX_PATH] = {0};
|
|
wsprintf(search, "%s\\*.*", buf);
|
|
HANDLE files = FindFirstFile(search, &data);
|
|
if (files == INVALID_HANDLE_VALUE) return ; // nothin' in it
|
|
for (;;)
|
|
{
|
|
wchar_t obuf[WA_MAX_PATH] = {0};
|
|
swprintf(obuf, L"%s\\%s", buf, data.cFileName);
|
|
addDroppedFile(obuf, plist);
|
|
if (!FindNextFile(files, &data))
|
|
{
|
|
FindClose(files);
|
|
return ;
|
|
}
|
|
}
|
|
|
|
// should never get here
|
|
}
|
|
else
|
|
{
|
|
addDragItem(DD_FILENAME, plist->addItem(new FilenamePS(StringPrintfW(L"file:%s", buf))));
|
|
}
|
|
#else
|
|
addDragItem(DD_FILENAME, plist->addItem(new FilenamePS(StringPrintfW(L"file:%s", filename))));
|
|
#endif
|
|
}
|
|
|
|
bool BaseWnd::getNoCopyBits(void)
|
|
{
|
|
return ncb;
|
|
}
|
|
|
|
void BaseWnd::setNoCopyBits(bool newncb)
|
|
{
|
|
ncb = newncb;
|
|
}
|
|
|
|
int BaseWnd::onEraseBkgnd(HDC dc)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::setIcon(OSICONHANDLE icon, int _small)
|
|
{
|
|
Wasabi::Std::Wnd::setIcon(getOsWindowHandle(), icon, !_small);
|
|
//CUT SendMessageW(getOsWindowHandle(), WM_SETICON, _small ? ICON_SMALL : ICON_BIG, (int)icon);
|
|
}
|
|
|
|
const wchar_t *BaseWnd::getTip()
|
|
{
|
|
return tip;
|
|
}
|
|
|
|
void BaseWnd::setTip(const wchar_t *_tooltip)
|
|
{
|
|
tip = _tooltip;
|
|
abortTip();
|
|
}
|
|
|
|
int BaseWnd::getStartHidden()
|
|
{
|
|
return start_hidden;
|
|
}
|
|
|
|
void BaseWnd::abortTip()
|
|
{
|
|
if (tipshowtimer)
|
|
{
|
|
// TODO: on the mac, use CreateMouseTrackingRegion
|
|
TRACKMOUSEEVENT tracker;
|
|
tracker.cbSize=sizeof(tracker);
|
|
tracker.dwFlags = TME_HOVER|TME_CANCEL;
|
|
tracker.hwndTrack = this->getOsWindowHandle();
|
|
tracker.dwHoverTime = TIP_TIMER_THRESHOLD;
|
|
|
|
TrackMouseEvent(&tracker);
|
|
}
|
|
tipshowtimer = FALSE;
|
|
if (tipawaytimer)
|
|
killTimer(TIP_AWAY_ID);
|
|
tipawaytimer = FALSE;
|
|
if (tipdestroytimer)
|
|
killTimer(TIP_DESTROYTIMER_ID);
|
|
tipdestroytimer = FALSE;
|
|
destroyTip();
|
|
tip_done = FALSE;
|
|
RECT r;
|
|
if (getOsWindowHandle() && Wasabi::Std::Wnd::getUpdateRect(getOsWindowHandle(), &r) != 0) // FG> avoids xoring over disapearing tip
|
|
repaint();
|
|
}
|
|
|
|
int BaseWnd::isVisible(int within)
|
|
{
|
|
if (!isVirtual() && !getOsWindowHandle()) return 0;
|
|
if (!this_visible) return 0;
|
|
|
|
if (within) return this_visible; // whatever, local
|
|
|
|
if (isVirtual()) // virtual, global
|
|
if (getParent())
|
|
return getParent()->isVisible();
|
|
else
|
|
return 0;
|
|
|
|
// non virtual, global
|
|
if (Wasabi::Std::Wnd::isPopup(getOsWindowHandle())) return this_visible;
|
|
if (!Wasabi::Std::Wnd::isValidWnd(Wasabi::Std::Wnd::getParent(getOsWindowHandle()))) return 0;
|
|
if (getParent()) return getParent()->isVisible(); // not a popup, check its parent or fail
|
|
return this_visible;
|
|
}
|
|
|
|
void BaseWnd::registerRootWndChild(ifc_window *child)
|
|
{
|
|
rootwndchildren.addItem(child);
|
|
if (child->isVirtual())
|
|
virtualChildren.addItem(child);
|
|
}
|
|
|
|
void BaseWnd::unregisterRootWndChild(ifc_window *child)
|
|
{
|
|
delTabOrderEntry(child);
|
|
rootwndchildren.removeItem(child);
|
|
if (curVirtualChildCaptured == child)
|
|
{
|
|
setVirtualChildCapture(NULL);
|
|
}
|
|
if (curVirtualChildFocus == child)
|
|
curVirtualChildFocus = NULL;
|
|
virtualChildren.removeItem(child);
|
|
//WASABI_API_WND->timer_remove(this, -1);
|
|
if (isPostOnInit() && isVisible())
|
|
postDeferredCallback(DEFERREDCB_INVALIDATE, 0);
|
|
}
|
|
|
|
int BaseWnd::isVirtual()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
//CUT?inline int isInRect(RECT *r,int x,int y) {
|
|
//CUT? if (x>=r->left&&x<=r->right&&y>=r->top&&y<=r->bottom) return 1;
|
|
//CUT? return 0;
|
|
//CUT?}
|
|
|
|
int BaseWnd::ensureVirtualCanvasOk()
|
|
{
|
|
RECT ncr;
|
|
|
|
if (isVirtual() && getParent()) return 1;
|
|
|
|
int size_w = rwidth;
|
|
int size_h = rheight;
|
|
|
|
if (!size_w || !size_h) return 0;
|
|
|
|
if (!virtualCanvas || virtualCanvasH != size_h || virtualCanvasW != size_w)
|
|
{
|
|
if (virtualCanvas)
|
|
{
|
|
deleteFrameBuffer(virtualCanvas);
|
|
virtualCanvas = NULL;
|
|
}
|
|
delete scalecanvas;
|
|
scalecanvas = NULL;
|
|
virtualCanvas = createFrameBuffer(size_w, size_h);
|
|
prepareFrameBuffer(virtualCanvas, size_w, size_h);
|
|
virtualCanvas->setBaseWnd(this);
|
|
virtualCanvasH = size_h; virtualCanvasW = size_w;
|
|
BaseWnd::getNonClientRect(&ncr);
|
|
deferedInvalidateRect(&ncr);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
Canvas *BaseWnd::createFrameBuffer(int w, int h)
|
|
{
|
|
return new BltCanvas(w, h, getOsWindowHandle());
|
|
}
|
|
|
|
void BaseWnd::prepareFrameBuffer(Canvas *canvas, int w, int h)
|
|
{
|
|
RECT r = {0, 0, w, h};
|
|
RegionI reg(&r);
|
|
virtualBeforePaint(®);
|
|
#ifdef _WIN32
|
|
canvas->selectClipRgn(NULL);
|
|
canvas->fillRect(&r, 0);
|
|
#elif defined(__APPLE__)
|
|
CGContextClearRect(canvas->getHDC(), CGRectInfinite); // TODO: make "clear" function in canvas
|
|
#endif
|
|
virtualAfterPaint(®);
|
|
}
|
|
|
|
void BaseWnd::deleteFrameBuffer(Canvas *canvas)
|
|
{
|
|
delete canvas;
|
|
}
|
|
|
|
// paints the client content, followed by the virtual child tree. recursive
|
|
int BaseWnd::paintTree(Canvas *canvas, api_region *r)
|
|
{
|
|
onPaint(canvas, r);
|
|
|
|
#ifdef WASABI_DRAW_FOCUS_RECT
|
|
if (gotFocus())
|
|
{
|
|
RECT ncr;
|
|
getNonClientRect(&ncr);
|
|
// try to use skinned focus rect
|
|
if (WASABI_API_WND->paintset_present(Paintset::FOCUSRECT))
|
|
WASABI_API_WND->paintset_render(Paintset::FOCUSRECT, canvas, &ncr, 128);
|
|
else // otherwise this looks kinda nice :P
|
|
canvas->drawRect(&ncr, 0, 0xFFFFFF, 128);
|
|
}
|
|
#endif
|
|
|
|
if (isVirtual() && !hasVirtualChildren()) return 0;
|
|
|
|
api_region *hostrgn = NULL;
|
|
api_region *update = r;
|
|
|
|
if (!(hwnd != NULL && getParent() == NULL))
|
|
{
|
|
hostrgn = getRegion();
|
|
update = r->clone();
|
|
if (hostrgn && !isRectRgn())
|
|
{
|
|
RECT ncr = clientRect();
|
|
api_region *hostclone = hostrgn->clone();
|
|
hostclone->addRegion(getComposedRegion());
|
|
hostclone->offset(ncr.left, ncr.top);
|
|
update->andRegion(hostclone);
|
|
hostrgn->disposeClone(hostclone);
|
|
}
|
|
}
|
|
|
|
RegionI client_update;
|
|
for (int i = 0;i < virtualChildren.getNumItems();i++)
|
|
{
|
|
if (!virtualChildren[i]->isVisible(1)) continue;
|
|
RECT rChild;
|
|
ifc_window *w = virtualChildren[i];
|
|
w->getNonClientRect(&rChild);
|
|
if ((rChild.right != rChild.left) && (rChild.bottom != rChild.top))
|
|
if (update->intersectRect(&rChild, &client_update))
|
|
{
|
|
w->paintTree(canvas, &client_update);
|
|
}
|
|
}
|
|
|
|
if (update != r) r->disposeClone(update);
|
|
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::setVirtualCanvas(Canvas *c)
|
|
{
|
|
virtualCanvas = c;
|
|
}
|
|
|
|
int BaseWnd::pointInWnd(POINT *p)
|
|
{
|
|
RECT r;
|
|
if (!isVisible(1)) return 0;
|
|
getWindowRect(&r);
|
|
if (!Wasabi::Std::pointInRect(r, *p))
|
|
return 0;
|
|
for (int i = 0; i < getNumRootWndChildren(); i++)
|
|
{
|
|
ifc_window *c = enumRootWndChildren(i);
|
|
if (!c->isVisible(1)) continue;
|
|
RECT rChild;
|
|
c->getWindowRect(&rChild);
|
|
if (Wasabi::Std::pointInRect(rChild, *p))
|
|
return 0;
|
|
}
|
|
//NONPORTABLE
|
|
/* HWND child = GetWindow(getOsWindowHandle(), GW_CHILD);
|
|
while (child != NULL) {
|
|
if (IsWindowVisible(child)) {
|
|
RECT r2;
|
|
GetWindowRect(child, &r2);
|
|
if (Std::pointInRect(r2, *p))
|
|
return 0;
|
|
}
|
|
child = GetWindow(child, GW_HWNDNEXT);
|
|
}*/
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::paint(Canvas *c, api_region *r)
|
|
{
|
|
if (isVirtual())
|
|
{
|
|
RegionI d;
|
|
RECT cr;
|
|
getClientRect(&cr);
|
|
if (r == NULL)
|
|
{
|
|
d.addRect(&cr);
|
|
}
|
|
else
|
|
{
|
|
d.addRegion(r);
|
|
d.offset(cr.left, cr.top);
|
|
}
|
|
ifc_window *rp = getRootParent();
|
|
deferedInvalidate();
|
|
rp->paint(NULL, &d);
|
|
BltCanvas *cc = static_cast<BltCanvas*>(rp->getFrameBuffer());
|
|
if (r != NULL) c->selectClipRgn(r);
|
|
cc->blit(cr.left, cr.top, c, 0, 0, cr.right - cr.left, cr.bottom - cr.top);
|
|
return 1;
|
|
}
|
|
|
|
if (!ensureVirtualCanvasOk()) return 0;
|
|
RegionI *deleteme = NULL;
|
|
if (r == NULL)
|
|
{
|
|
RECT cr;
|
|
getNonClientRect(&cr);
|
|
deleteme = new RegionI(&cr);
|
|
r = deleteme;
|
|
}
|
|
|
|
virtualBeforePaint(r);
|
|
|
|
RECT rcPaint;
|
|
r->getBox(&rcPaint);
|
|
|
|
double ra = getRenderRatio();
|
|
|
|
if (deferedInvalidRgn)
|
|
{
|
|
api_region *nr = NULL;
|
|
if (renderRatioActive())
|
|
{
|
|
nr = r->clone();
|
|
double d = 1.0 / ra;
|
|
nr->scale(d, d, TRUE);
|
|
}
|
|
|
|
if (deferedInvalidRgn->isEmpty() == 0)
|
|
{
|
|
// some deferednvalidated regions needs to be repainted
|
|
// TODO: need a "clear region" function in canvas
|
|
api_region *i = deferedInvalidRgn->clone();
|
|
#ifdef _WIN32
|
|
FillRgn(virtualCanvas->getHDC(), i->getOSHandle(), (HBRUSH)GetStockObject(BLACK_BRUSH));
|
|
#elif defined(__APPLE__)
|
|
CGContextSaveGState(virtualCanvas->getHDC());
|
|
virtualCanvas->selectClipRgn(i);
|
|
CGContextClearRect(virtualCanvas->getHDC(), CGRectInfinite);
|
|
// virtualCanvas->selectClipRgn(0);
|
|
CGContextRestoreGState(virtualCanvas->getHDC());
|
|
#endif
|
|
paintTree(virtualCanvas, i);
|
|
|
|
deferedValidateRgn(i);
|
|
deferedInvalidRgn->disposeClone(i);
|
|
|
|
}
|
|
if (nr) r->disposeClone(nr);
|
|
}
|
|
|
|
virtualAfterPaint(r);
|
|
|
|
if (c != NULL)
|
|
{
|
|
commitFrameBuffer(c, &rcPaint, ra); //TH WDP2-212
|
|
}
|
|
|
|
delete deleteme;
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::virtualOnPaint()
|
|
{
|
|
#ifdef _WIN32
|
|
RECT cr;
|
|
getNonClientRect(&cr);
|
|
if (cr.left >= cr.right || cr.top >= cr.bottom) return 0;
|
|
|
|
if (!ensureVirtualCanvasOk()) return 0;
|
|
|
|
RegionI reg;
|
|
|
|
//CUT GetUpdateRgn(getOsWindowHandle(), r->getOSHandle(), FALSE);
|
|
Wasabi::Std::Wnd::getUpdateRegion(getOsWindowHandle(), reg.getOSHandle());
|
|
|
|
PaintCanvas paintcanvas;
|
|
if (!paintcanvas.beginPaint(this))
|
|
{
|
|
virtualAfterPaint(®); return 0;
|
|
}
|
|
|
|
// DO NOT DELETE - This looks like it does nothing, but it actually makes the GDI call us again with WM_PAINT if some window
|
|
// moves over this one between BeginPaint and EndPaint. We still use GetUpdateRgn so we don't have to check for
|
|
// the version of Windows. See doc. If this function is not available (should be here in 95/98/NT/2K, but we never know)
|
|
// then we use the rcPaint rect... less precise, but still works.
|
|
//CUT if (getRandomRgn) {
|
|
if (Wasabi::Std::Wnd::haveGetRandomRegion())
|
|
{
|
|
RegionI zap;
|
|
//CUT getRandomRgn(paintcanvas.getHDC(), zap.getOSHandle(), SYSRGN);
|
|
Wasabi::Std::Wnd::getRandomRegion(paintcanvas.getHDC(), zap.getOSHandle());
|
|
}
|
|
else
|
|
{
|
|
RECT z;
|
|
paintcanvas.getRcPaint(&z);
|
|
reg.setRect(&z);
|
|
}
|
|
|
|
// -------------
|
|
|
|
/*// for debug
|
|
HDC dc = GetDC(getOsWindowHandle());
|
|
InvertRgn(dc, r->getHRGN());
|
|
InvertRgn(dc, r->getHRGN());
|
|
ReleaseDC(getOsWindowHandle(), dc);*/
|
|
|
|
paint(&paintcanvas, ®);
|
|
#else
|
|
#warning port me or remove me
|
|
#endif
|
|
|
|
return 1;
|
|
}
|
|
|
|
ifc_window *BaseWnd::enumVirtualChild(int _enum)
|
|
{
|
|
return virtualChildren[_enum];
|
|
}
|
|
|
|
int BaseWnd::getNumVirtuals()
|
|
{
|
|
return virtualChildren.getNumItems();
|
|
}
|
|
|
|
ifc_window *BaseWnd::enumRootWndChildren(int n)
|
|
{
|
|
return rootwndchildren.enumItem(n);
|
|
}
|
|
|
|
int BaseWnd::getNumRootWndChildren()
|
|
{
|
|
return rootwndchildren.getNumItems();
|
|
}
|
|
|
|
ifc_window *BaseWnd::findRootWndChild(int x, int y, int only_virtuals)
|
|
{
|
|
for (int i = getNumRootWndChildren() - 1; i > -1; i--)
|
|
{
|
|
RECT r;
|
|
ifc_window *child = enumRootWndChildren(i);
|
|
//DebugStringW(L"findRootWndChild = entering = %s\n", child->getId());
|
|
if (only_virtuals && !child->isVirtual()) continue;
|
|
child->getNonClientRect(&r);
|
|
int _x = x;
|
|
int _y = y;
|
|
if (!child->isVirtual())
|
|
{
|
|
POINT pt;
|
|
child->getPosition(&pt);
|
|
_x -= pt.x;
|
|
_y -= pt.y;
|
|
}
|
|
int iv = child->isVisible(1);
|
|
//int gpa = child->getPaintingAlpha();
|
|
POINT _p = Wasabi::Std::makePoint(_x, _y);
|
|
if (iv /*&& gpa > 0*/ && Wasabi::Std::pointInRect(r, _p))
|
|
{
|
|
// GROUP
|
|
ifc_window *z = child->findRootWndChild(_x, _y);
|
|
if (z) return z;
|
|
}
|
|
/*gpa > 0 &&*/
|
|
/*if (iv && _x>=r.left&&_x<=r.right&&_y>=r.top&&_y<=r.bottom && !child->isClickThrough() && child->ptInRegion(_x, _y)) {
|
|
return child;
|
|
}*/
|
|
}
|
|
return (!isClickThrough() && ptInRegion(x, y)) ? this : NULL;
|
|
}
|
|
|
|
//PORTME
|
|
int BaseWnd::handleVirtualChildMsg(UINT uMsg, int x, int y, void *p, void *d)
|
|
{
|
|
#ifdef _WIN32
|
|
ifc_window *child = NULL;
|
|
|
|
if (curVirtualChildCaptured)
|
|
child = curVirtualChildCaptured;
|
|
else
|
|
child = findRootWndChild(x, y, 1); // warning, can return this which is not virtual
|
|
|
|
// ASSERT(child != NULL); // BU this came up and I don't know why, looks like it should never happen
|
|
// FG> actually it can happen when coming back from a popup menu when cpu usage is high, the window might be
|
|
// hidden (destroying) and ptInRegion returns false.
|
|
if (!child) return 0;
|
|
|
|
//int isvirtual = child->isVirtual();
|
|
|
|
if (child) child = child->getForwardWnd();
|
|
|
|
if (child && child->isEnabled())
|
|
{
|
|
switch (uMsg)
|
|
{
|
|
case WM_LBUTTONDOWN:
|
|
/* if (isvirtual && child != curVirtualChildFocus)
|
|
focusVirtualChild(child);*/
|
|
autoFocus(child);
|
|
if (child->wantLeftClicks())
|
|
child->onLeftButtonDown(x, y);
|
|
if (child->checkDoubleClick(uMsg, x, y) && child->wantDoubleClicks())
|
|
child->onLeftButtonDblClk(x, y);
|
|
return 1;
|
|
case WM_RBUTTONDOWN:
|
|
/* if (isvirtual && child != curVirtualChildFocus)
|
|
focusVirtualChild(child);*/
|
|
autoFocus(child);
|
|
if (child->wantRightClicks())
|
|
child->onRightButtonDown(x, y);
|
|
if (child->checkDoubleClick(uMsg, x, y) && child->wantDoubleClicks())
|
|
child->onRightButtonDblClk(x, y);
|
|
return 1;
|
|
case WM_LBUTTONUP:
|
|
if (child->wantLeftClicks())
|
|
child->onLeftButtonUp(x, y);
|
|
return 1;
|
|
case WM_RBUTTONUP:
|
|
if (child->wantRightClicks())
|
|
child->onRightButtonUp(x, y);
|
|
return 1;
|
|
case WM_MOUSEMOVE:
|
|
{
|
|
if (curVirtualChildCaptured == child || (curVirtualChildCaptured == NULL))
|
|
{
|
|
if (child->wantMouseMoves())
|
|
child->onMouseMove(x, y);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
case WM_MOUSEHOVER:
|
|
((BaseWnd *)child)->onTip();
|
|
break;
|
|
case WM_SETCURSOR:
|
|
int a = child->getCursorType(x, y);
|
|
if (!p) return 0;
|
|
*(int *)p = a;
|
|
if (a == BASEWND_CURSOR_USERSET)
|
|
{
|
|
OSCURSORHANDLE c = child->getCustomCursor(x, y);
|
|
if (!d) return 0;
|
|
*(OSCURSORHANDLE *)d = c;
|
|
}
|
|
return 1;
|
|
}
|
|
}
|
|
#else
|
|
#warning port me or remove me
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onLeftButtonDown(int x, int y)
|
|
{
|
|
disable_tooltip_til_recapture = 1;
|
|
abortTip();
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onLeftButtonUp(int x, int y)
|
|
{
|
|
disable_tooltip_til_recapture = 1;
|
|
abortTip();
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::setVirtualChildCapture(ifc_window *child)
|
|
{
|
|
if (child)
|
|
{
|
|
if (!inputCaptured)
|
|
{
|
|
beginCapture();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
endCapture();
|
|
}
|
|
curVirtualChildCaptured = child;
|
|
}
|
|
|
|
ifc_window *BaseWnd::getVirtualChildCapture()
|
|
{
|
|
if (inputCaptured && Wasabi::Std::Wnd::getCapture() == getOsWindowHandle())
|
|
return curVirtualChildCaptured;
|
|
else
|
|
if (inputCaptured) inputCaptured = 0;
|
|
return NULL;
|
|
}
|
|
|
|
ifc_window *BaseWnd::getBaseTextureWindow()
|
|
{
|
|
// return our base texture window if we have it
|
|
if (btexture)
|
|
return btexture;
|
|
// return our parent's if they have it
|
|
if (getParent())
|
|
return getParent()->getBaseTextureWindow();
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
void BaseWnd::renderBaseTexture(ifc_canvas *c, const RECT &r, int alpha)
|
|
{
|
|
WASABI_API_WND->skin_renderBaseTexture(getBaseTextureWindow(), c, r, this, alpha);
|
|
}
|
|
|
|
void BaseWnd::setBaseTextureWindow(ifc_window *w)
|
|
{
|
|
btexture = w;
|
|
}
|
|
|
|
void BaseWnd::setNotifyWindow(ifc_window *newnotify)
|
|
{
|
|
notifyWindow = newnotify;
|
|
}
|
|
|
|
ifc_window *BaseWnd::getNotifyWindow()
|
|
{
|
|
return destroying ? NULL : notifyWindow;
|
|
}
|
|
|
|
int BaseWnd::gotFocus()
|
|
{
|
|
return hasfocus && curVirtualChildFocus == NULL;
|
|
}
|
|
|
|
int BaseWnd::isActive()
|
|
{
|
|
OSWINDOWHANDLE h = hwnd;
|
|
if (h == NULL)
|
|
{
|
|
ifc_window *par = getParent();
|
|
if (par == NULL) return 0;
|
|
h = par->getOsWindowHandle();
|
|
}
|
|
if (h == NULL) return 0;
|
|
return (Wasabi::Std::Wnd::getActiveWindow() == h);
|
|
}
|
|
|
|
int BaseWnd::onChar(unsigned int c)
|
|
{
|
|
switch (c)
|
|
{
|
|
case 9: // TAB
|
|
if (Std::keyModifier(STDKEY_SHIFT))
|
|
focusPrevious();
|
|
else
|
|
focusNext();
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*int BaseWnd::focusVirtualChild(api_window *child) {
|
|
if (!gotFocus()) setFocus();
|
|
if (!child->wantFocus()) return 0;
|
|
setVirtualChildFocus(child);
|
|
return 1;
|
|
}*/
|
|
|
|
int BaseWnd::wantFocus()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
// Return 1 if there is a modal window that is not this
|
|
int BaseWnd::checkModal()
|
|
{
|
|
if (bypassModal()) return 0;
|
|
ifc_window *w = WASABI_API_WND->getModalWnd();
|
|
if (w && w != static_cast<ifc_window*>(this) && w != getDesktopParent())
|
|
{
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::cascadeRepaintFrom(ifc_window *who, int pack)
|
|
{
|
|
RECT r;
|
|
BaseWnd::getNonClientRect(&r);
|
|
return BaseWnd::cascadeRepaintRect(&r, pack);
|
|
}
|
|
|
|
int BaseWnd::cascadeRepaint(int pack)
|
|
{
|
|
return cascadeRepaintFrom(this, pack);
|
|
}
|
|
|
|
int BaseWnd::cascadeRepaintRgn(api_region *r, int pack)
|
|
{
|
|
return cascadeRepaintRgnFrom(r, this, pack);
|
|
}
|
|
|
|
int BaseWnd::cascadeRepaintRect(RECT *r, int pack)
|
|
{
|
|
return cascadeRepaintRectFrom(r, this, pack);
|
|
}
|
|
|
|
int BaseWnd::cascadeRepaintRectFrom(RECT *r, ifc_window *who, int pack)
|
|
{
|
|
RegionI reg(r);
|
|
int rt = cascadeRepaintRgnFrom(®, who, pack);
|
|
return rt;
|
|
}
|
|
|
|
void BaseWnd::_cascadeRepaintRgn(api_region *rg)
|
|
{
|
|
if (!ensureVirtualCanvasOk()) return ;
|
|
|
|
WndCanvas paintcanvas;
|
|
if (paintcanvas.attachToClient(this) == 0)
|
|
return;
|
|
|
|
virtualBeforePaint(rg);
|
|
|
|
deferedInvalidateRgn(rg);
|
|
paintTree(virtualCanvas, rg);
|
|
|
|
virtualAfterPaint(rg);
|
|
|
|
double ra = getRenderRatio();
|
|
|
|
RECT rcPaint;
|
|
rg->getBox(&rcPaint);
|
|
|
|
RECT rc;
|
|
getClientRect(&rc); //JF> this gets it in virtual (non-scaled) coordinates,
|
|
// so we need to do these comparisons before scaling.
|
|
rcPaint.bottom = MIN((int)rc.bottom, (int)rcPaint.bottom);
|
|
rcPaint.right = MIN((int)rc.right, (int)rcPaint.right);
|
|
|
|
if (renderRatioActive()) // probably faster than scaling the clone
|
|
{
|
|
rcPaint.left = (int)((rcPaint.left - 1) * ra);
|
|
rcPaint.top = (int)((rcPaint.top - 1) * ra);
|
|
rcPaint.right = (int)(rcPaint.right * ra + 0.999999);
|
|
rcPaint.bottom = (int)(rcPaint.bottom * ra + 0.999999);
|
|
}
|
|
rcPaint.left = MAX(0, (int)rcPaint.left);
|
|
rcPaint.top = MAX(0, (int)rcPaint.top);
|
|
rcPaint.right = MIN((int)rcPaint.right, (int)(rwidth * ra));
|
|
rcPaint.bottom = MIN((int)rcPaint.bottom, (int)(rheight * ra));
|
|
|
|
commitFrameBuffer(&paintcanvas, &rcPaint, ra);
|
|
}
|
|
|
|
|
|
void BaseWnd::packCascadeRepaintRgn(api_region *rg)
|
|
{
|
|
if (!deferedCascadeRepaintRgn) deferedCascadeRepaintRgn = new RegionI;
|
|
deferedCascadeRepaintRgn->addRegion(rg);
|
|
need_flush_cascaderepaint = 1;
|
|
}
|
|
|
|
int BaseWnd::cascadeRepaintRgnFrom(api_region *_rg, ifc_window *who, int pack)
|
|
{
|
|
|
|
api_region *rg = _rg->clone();
|
|
|
|
int j = virtualChildren.searchItem(who);
|
|
for (int i = 0; i < virtualChildren.getNumItems(); i++)
|
|
{
|
|
ifc_window *w = virtualChildren[i];
|
|
if (w != who && w->wantSiblingInvalidations())
|
|
w->onSiblingInvalidateRgn(rg, who, j, i);
|
|
}
|
|
|
|
if (!pack)
|
|
{
|
|
_cascadeRepaintRgn(rg);
|
|
}
|
|
else
|
|
{
|
|
packCascadeRepaintRgn(rg);
|
|
}
|
|
|
|
_rg->disposeClone(rg);
|
|
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::setDesktopAlpha(int a)
|
|
{
|
|
if (a && !Wasabi::Std::Wnd::isDesktopAlphaAvailable()) return ;
|
|
if (a == w2k_alpha) return ;
|
|
w2k_alpha = a;
|
|
if (!a && scalecanvas)
|
|
{
|
|
delete scalecanvas;
|
|
scalecanvas = NULL;
|
|
}
|
|
setLayeredWindow(w2k_alpha);
|
|
onSetDesktopAlpha(a);
|
|
}
|
|
|
|
void BaseWnd::onSetDesktopAlpha(int a) { }
|
|
|
|
void BaseWnd::commitFrameBuffer(Canvas *paintcanvas, RECT *r, double ra)
|
|
{
|
|
if (w2k_alpha && Wasabi::Std::Wnd::isDesktopAlphaAvailable() && !cloaked)
|
|
{
|
|
//CUT BLENDFUNCTION blend= {AC_SRC_OVER, 0, wndalpha, AC_SRC_ALPHA };
|
|
//CUT POINT pt={0,0};
|
|
RECT spr;
|
|
getWindowRect(&spr);
|
|
//CUT POINT sp={spr.left,spr.top};
|
|
//CUT SIZE ss={spr.right-spr.left, spr.bottom-spr.top};
|
|
int sw = spr.right - spr.left, sh = spr.bottom - spr.top;
|
|
//CUT SysCanvas c;
|
|
|
|
if (handleRatio() && renderRatioActive())
|
|
{
|
|
// eek slow!
|
|
RECT r;
|
|
getWindowRect(&r);
|
|
int w = r.right - r.left;
|
|
int h = r.bottom - r.top;
|
|
if (!scalecanvas) scalecanvas = new BltCanvas(w, h, getOsWindowHandle());
|
|
virtualCanvas->stretchblit(0, 0, (int)((double)virtualCanvasW * 65536.0), (int)((double)virtualCanvasH * 65536.0), scalecanvas, 0, 0, w, h);
|
|
}
|
|
|
|
//CUT updateLayeredWindow(hwnd, c.getHDC(), &sp, &ss, (scalecanvas ? scalecanvas : virtualCanvas)->getHDC(), &pt, 0, &blend, ULW_ALPHA);
|
|
Wasabi::Std::Wnd::updateLayeredWnd(hwnd, spr.left, spr.top, sw, sh, (scalecanvas ? scalecanvas : virtualCanvas)->getHDC(), wndalpha);
|
|
}
|
|
else
|
|
{
|
|
if (ABS(ra - 1.0) <= 0.01)
|
|
{
|
|
virtualCanvas->blit(r->left, r->top, paintcanvas, r->left, r->top, r->right - r->left, r->bottom - r->top);
|
|
}
|
|
else
|
|
{
|
|
RECT tr = *r;
|
|
|
|
double invra = 65536.0 / ra;
|
|
int lp = tr.left;
|
|
int tp = tr.top;
|
|
int w = tr.right - tr.left;
|
|
int h = tr.bottom - tr.top;
|
|
|
|
int sx = (int)((double)lp * invra);
|
|
int sy = (int)((double)tp * invra);
|
|
int sw = (int)((double)w * invra);
|
|
int sh = (int)((double)h * invra);
|
|
|
|
virtualCanvas->stretchblit(sx, sy, sw, sh, paintcanvas, lp, tp, w, h);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BaseWnd::flushPaint()
|
|
{
|
|
postDeferredCallback(DEFERREDCB_FLUSHPAINT, 0);
|
|
}
|
|
|
|
void BaseWnd::do_flushPaint()
|
|
{
|
|
if (!deferedInvalidRgn || deferedInvalidRgn->isEmpty()) return ;
|
|
api_region *r = deferedInvalidRgn->clone();
|
|
cascadeRepaintRgn(r);
|
|
deferedInvalidRgn->disposeClone(r);
|
|
deferedInvalidRgn->empty();
|
|
}
|
|
|
|
int BaseWnd::isMouseOver(int x, int y)
|
|
{
|
|
POINT pt = {x, y};
|
|
clientToScreen(&pt);
|
|
|
|
return (WASABI_API_WND->rootWndFromPoint(&pt) == this);
|
|
}
|
|
|
|
void BaseWnd::freeResources()
|
|
{}
|
|
|
|
void BaseWnd::reloadResources()
|
|
{
|
|
invalidate();
|
|
}
|
|
|
|
double BaseWnd::getRenderRatio()
|
|
{
|
|
if (!handleRatio()) return 1.0;
|
|
if (!ratiolinked) return ratio;
|
|
return getParent() ? getParent()->getRenderRatio() : ratio;
|
|
}
|
|
|
|
void BaseWnd::setRenderRatio(double r)
|
|
{
|
|
// "snap" to 1.0
|
|
if (ABS(r - 1.0) <= 0.02f) r = 1.0;
|
|
if (scalecanvas)
|
|
{
|
|
delete scalecanvas;
|
|
scalecanvas = NULL;
|
|
}
|
|
if (isInited() && r != ratio && !isVirtual() && (!getParent() || !ratiolinked))
|
|
{
|
|
// must scale size & region accordingly
|
|
RECT rc;
|
|
BaseWnd::getWindowRect(&rc);
|
|
rc.right = rc.left + rwidth;
|
|
rc.bottom = rc.top + rheight;
|
|
ratio = r;
|
|
|
|
resize(&rc);
|
|
|
|
invalidate();
|
|
if (isPostOnInit())
|
|
onRatioChanged();
|
|
}
|
|
}
|
|
|
|
void BaseWnd::setRatioLinked(int l)
|
|
{
|
|
ratiolinked = l;
|
|
if (isPostOnInit())
|
|
setRenderRatio(ratio);
|
|
}
|
|
|
|
int BaseWnd::renderRatioActive()
|
|
{
|
|
return ABS(getRenderRatio() - 1.0) > 0.01f;
|
|
}
|
|
|
|
void BaseWnd::multRatio(int *x, int *y)
|
|
{
|
|
double rr = getRenderRatio();
|
|
if (x) *x = (int)((double)(*x) * rr);
|
|
if (y) *y = (int)((double)(*y) * rr);
|
|
}
|
|
|
|
void BaseWnd::multRatio(RECT *r)
|
|
{
|
|
Wasabi::Std::scaleRect(r, getRenderRatio());
|
|
}
|
|
|
|
void BaseWnd::divRatio(int *x, int *y)
|
|
{
|
|
double rr = getRenderRatio();
|
|
if (x) *x = (int)((double)(*x) / rr + 0.5);
|
|
if (y) *y = (int)((double)(*y) / rr + 0.5);
|
|
}
|
|
|
|
void BaseWnd::divRatio(RECT *r)
|
|
{
|
|
double rr = getRenderRatio();
|
|
Wasabi::Std::scaleRect(r, 1./rr);
|
|
}
|
|
|
|
void BaseWnd::bringVirtualToFront(ifc_window *w)
|
|
{
|
|
changeChildZorder(w, 0);
|
|
}
|
|
|
|
void BaseWnd::bringVirtualToBack(ifc_window *w)
|
|
{
|
|
changeChildZorder(w, virtualChildren.getNumItems()-1);
|
|
}
|
|
|
|
void BaseWnd::bringVirtualAbove(ifc_window *w, ifc_window *b)
|
|
{
|
|
ASSERT(b->isVirtual());
|
|
int p = virtualChildren.searchItem(b);
|
|
if (p == -1) return ;
|
|
changeChildZorder(w, p);
|
|
}
|
|
|
|
void BaseWnd::bringVirtualBelow(ifc_window *w, ifc_window *b)
|
|
{
|
|
ASSERT(b->isVirtual());
|
|
int p = virtualChildren.searchItem(b);
|
|
if (p == -1) return ;
|
|
changeChildZorder(w, p + 1);
|
|
}
|
|
|
|
void BaseWnd::changeChildZorder(ifc_window *w, int newpos)
|
|
{
|
|
int p = newpos;
|
|
p = MAX(p, (int)0);
|
|
p = MIN(p, virtualChildren.getNumItems()-1);
|
|
RECT cr;
|
|
w->getClientRect(&cr);
|
|
|
|
PtrList<ifc_window> l;
|
|
int i;
|
|
for (i = 0;i < virtualChildren.getNumItems();i++)
|
|
if (virtualChildren[i] != w)
|
|
l.addItem(virtualChildren[i]);
|
|
|
|
p = virtualChildren.getNumItems() - newpos - 1;
|
|
virtualChildren.removeAll();
|
|
|
|
int done = 0;
|
|
|
|
for (i = 0;i < l.getNumItems();i++)
|
|
if (i == p && !done)
|
|
{
|
|
virtualChildren.addItem(w);
|
|
i--;
|
|
done++;
|
|
}
|
|
else
|
|
{
|
|
RECT dr, intersection;
|
|
l.enumItem(i)->getClientRect(&dr);
|
|
if (Wasabi::Std::rectIntersect(intersection, dr, &cr))
|
|
l[i]->invalidateRect(&intersection);
|
|
virtualChildren.addItem(l.enumItem(i));
|
|
}
|
|
if (i == p && !done)
|
|
virtualChildren.addItem(w);
|
|
w->invalidate();
|
|
}
|
|
|
|
int BaseWnd::onActivate()
|
|
{
|
|
if (hasVirtualChildren())
|
|
{
|
|
int l = getNumVirtuals();
|
|
for (int i = 0; i < l; i++)
|
|
{
|
|
ifc_window *r = enumVirtualChild(i);
|
|
r->onActivate();
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::onDeactivate()
|
|
{
|
|
if (hasVirtualChildren())
|
|
{
|
|
int l = getNumVirtuals();
|
|
for (int i = 0; i < l; i++)
|
|
{
|
|
ifc_window *r = enumVirtualChild(i);
|
|
r->onDeactivate();
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::getDesktopAlpha()
|
|
{
|
|
return w2k_alpha;
|
|
}
|
|
|
|
api_region *BaseWnd::getRegion()
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
//CUT int BaseWnd::isTransparencyAvailable() {
|
|
//CUT #ifdef WIN32
|
|
//CUT #else
|
|
//CUT #pragma warning port me!
|
|
//CUT #endif
|
|
//CUT return 0;
|
|
//CUT }
|
|
|
|
int BaseWnd::handleTransparency()
|
|
{
|
|
return 1; // by default all windows handle transparency, only windows blitting directly on the SCREEN (if you blit directly on the DC it's still ok),
|
|
} // for instance, a vis or a video using overlay should return 0, this will let the layout auto manage its alpha as that window is shown/hiden
|
|
|
|
void BaseWnd::setAlpha(int active, int inactive)
|
|
{
|
|
if (active == 254) active = 255;
|
|
if (active == 1) active = 0;
|
|
if (inactive == 254) inactive = 255;
|
|
if (inactive == 1) inactive = 0;
|
|
int oldactivealpha = activealpha;
|
|
active = MIN(255, MAX(0, active));
|
|
if (inactive != -1) inactive = MIN(255, MAX(0, inactive));
|
|
|
|
if (active != activealpha)
|
|
{
|
|
activealpha = active;
|
|
if (isActive())
|
|
{
|
|
invalidate();
|
|
if ((oldactivealpha == 0 || activealpha == 0) && (oldactivealpha != 0 || activealpha != 0))
|
|
invalidateWindowRegion();
|
|
}
|
|
}
|
|
if (inactive == -1) inactive = active;
|
|
if (inactive != inactivealpha)
|
|
{
|
|
inactivealpha = inactive;
|
|
if (!isActive())
|
|
{
|
|
invalidate();
|
|
if ((oldactivealpha == 0 || activealpha == 0) && (oldactivealpha != 0 || activealpha != 0))
|
|
invalidateWindowRegion();
|
|
}
|
|
}
|
|
}
|
|
|
|
void BaseWnd::getAlpha(int *active, int *inactive)
|
|
{
|
|
if (active) *active = activealpha;
|
|
if (inactive) *inactive = inactivealpha;
|
|
}
|
|
|
|
int BaseWnd::getPaintingAlpha(void)
|
|
{
|
|
int a = isActive() ? MIN(255, MAX(0, activealpha)) : MIN(255, MAX(0, inactivealpha));
|
|
ASSERT(a >= 0 && a <= 255);
|
|
if (getParent() && getParent()->isVirtual())
|
|
{
|
|
int b = getParent()->getPaintingAlpha();
|
|
a = (int)((double)a / 255.0 * (double)b);
|
|
}
|
|
if (a == 254) a = 255;
|
|
if (a == 1) a = 0;
|
|
if (!isEnabled()) a = (int)(a*0.6);
|
|
return a;
|
|
}
|
|
|
|
void BaseWnd::setClickThrough(int ct)
|
|
{
|
|
clickthrough = ct;
|
|
}
|
|
|
|
int BaseWnd::isClickThrough()
|
|
{
|
|
return clickthrough;
|
|
}
|
|
|
|
int BaseWnd::handleRatio()
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
#include <api/script/objects/c_script/c_rootobj.h>
|
|
|
|
int BaseWnd::createTip()
|
|
{
|
|
destroyTip();
|
|
tooltip = new Tooltip(getTip());
|
|
return -1;
|
|
}
|
|
|
|
void BaseWnd::destroyTip()
|
|
{
|
|
// this is to avoid pb if destroytip() is being called by a time while destroying tip
|
|
Tooltip *tt = tooltip;
|
|
tooltip = NULL;
|
|
delete tt;
|
|
}
|
|
|
|
|
|
int BaseWnd::runModal()
|
|
{
|
|
//PORTME
|
|
#ifdef _WIN32
|
|
ifc_window *dp = getDesktopParent();
|
|
if (dp && dp != this)
|
|
return dp->runModal();
|
|
|
|
MSG msg;
|
|
|
|
// SetCapture(NULL);
|
|
SetFocus(getOsWindowHandle());
|
|
|
|
WASABI_API_WND->pushModalWnd(this);
|
|
returnvalue = 0;
|
|
mustquit = 0;
|
|
|
|
// Main message loop:
|
|
while (!mustquit)
|
|
{
|
|
mustquit = !GetMessage(&msg, NULL, 0, 0);
|
|
if (!msg.hwnd || !TranslateAccelerator(msg.hwnd, NULL, &msg))
|
|
{
|
|
TranslateMessage(&msg);
|
|
DispatchMessage(&msg);
|
|
}
|
|
}
|
|
|
|
WASABI_API_WND->popModalWnd(this);
|
|
// SetCapture(NULL);
|
|
return returnvalue;
|
|
#else
|
|
#warning port me
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
void BaseWnd::endModal(int ret)
|
|
{
|
|
ifc_window *dp = getDesktopParent();
|
|
if (dp && dp != this)
|
|
{
|
|
dp->endModal(ret);
|
|
return ;
|
|
}
|
|
returnvalue = ret;
|
|
mustquit = 1;
|
|
}
|
|
|
|
int BaseWnd::wantAutoContextMenu()
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::onCancelCapture()
|
|
{}
|
|
|
|
ifc_window *BaseWnd::getNextVirtualFocus(ifc_window *w)
|
|
{
|
|
if (w == NULL)
|
|
{
|
|
if (childtabs.getNumItems() > 0)
|
|
return childtabs.getFirst()->wnd;
|
|
}
|
|
|
|
int a = getTabOrderEntry(w) + 1;
|
|
|
|
if (a < childtabs.getNumItems())
|
|
return childtabs.enumItem(a)->wnd;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
void BaseWnd::setVirtualChildFocus(ifc_window *w)
|
|
{
|
|
ASSERT(w && w->isVirtual());
|
|
if (curVirtualChildFocus)
|
|
curVirtualChildFocus->onKillFocus();
|
|
curVirtualChildFocus = w;
|
|
onSetRootFocus(w);
|
|
Wasabi::Std::Wnd::setFocus(getOsWindowHandle());
|
|
if (curVirtualChildFocus)
|
|
curVirtualChildFocus->onGetFocus();
|
|
}
|
|
|
|
int BaseWnd::ptInRegion(int x, int y)
|
|
{
|
|
RECT cr;
|
|
getNonClientRect(&cr);
|
|
POINT pt = {x - cr.left, y - cr.top};
|
|
api_region *reg = getRegion();
|
|
if (isRectRgn())
|
|
return (x >= cr.left && x <= cr.right && y >= cr.top && y <= cr.bottom);
|
|
return reg ? reg->ptInRegion(&pt) : 0;
|
|
}
|
|
|
|
api_region *BaseWnd::getComposedRegion()
|
|
{
|
|
ensureWindowRegionValid();
|
|
return composedrgn;
|
|
}
|
|
|
|
api_region *BaseWnd::getSubtractorRegion()
|
|
{
|
|
ensureWindowRegionValid();
|
|
return subtractorrgn;
|
|
}
|
|
|
|
int BaseWnd::ensureWindowRegionValid()
|
|
{
|
|
if (!isInited()) return 0;
|
|
if (wndregioninvalid)
|
|
{
|
|
computeComposedRegion();
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::invalidateWindowRegion()
|
|
{
|
|
wndregioninvalid = 1;
|
|
if (getParent()) getParent()->invalidateWindowRegion();
|
|
}
|
|
|
|
void BaseWnd::computeComposedRegion()
|
|
{
|
|
if (!isPostOnInit()) return ;
|
|
|
|
wndregioninvalid = 0;
|
|
|
|
RECT r;
|
|
getNonClientRect(&r);
|
|
|
|
api_region *reg = getRegion();
|
|
RegionI *_reg = NULL;
|
|
|
|
if (!reg)
|
|
{
|
|
_reg = new RegionI;
|
|
reg = _reg;
|
|
if (isRectRgn())
|
|
reg->setRect(&r);
|
|
}
|
|
else
|
|
if (isRectRgn())
|
|
reg->setRect(&r);
|
|
|
|
api_region *wr = reg->clone();
|
|
|
|
if (!subtractorrgn) subtractorrgn = new RegionI();
|
|
subtractorrgn->empty();
|
|
if (!composedrgn) composedrgn = new RegionI;
|
|
composedrgn->empty();
|
|
|
|
RegionI *subme = NULL;
|
|
RegionI *andme = NULL;
|
|
RegionI *orme = NULL;
|
|
|
|
// if subregion is now empty, we need to only use our region
|
|
RECT gr;
|
|
getNonClientRect(&gr);
|
|
for (int i = 0;i < virtualChildren.getNumItems();i++)
|
|
{
|
|
ifc_window *srw = virtualChildren.enumItem(i);
|
|
if (!srw->isVisible(1) || srw->getPaintingAlpha() == 0) continue;
|
|
if (srw->getRegionOp() != REGIONOP_NONE)
|
|
{
|
|
api_region *sr = srw->getComposedRegion();
|
|
if (sr)
|
|
{
|
|
api_region *osr = sr->clone();
|
|
RECT r;
|
|
srw->getNonClientRect(&r);
|
|
r.left -= gr.left;
|
|
r.top -= gr.top;
|
|
osr->offset(r.left, r.top);
|
|
/* sr->debug();
|
|
osr->debug();*/
|
|
if (srw->getRegionOp() == REGIONOP_OR)
|
|
{
|
|
if (!orme) orme = new RegionI;
|
|
orme->addRegion(osr);
|
|
}
|
|
else if (srw->getRegionOp() == REGIONOP_AND)
|
|
{
|
|
if (!andme) andme = new RegionI;
|
|
andme->addRegion(osr);
|
|
}
|
|
else if (srw->getRegionOp() == REGIONOP_SUB)
|
|
{
|
|
if (!subme) subme = new RegionI;
|
|
subme->addRegion(osr);
|
|
}
|
|
else if (srw->getRegionOp() == REGIONOP_SUB2)
|
|
{
|
|
if (!subme) subme = new RegionI;
|
|
subtractorrgn->addRegion(osr);
|
|
}
|
|
sr->disposeClone(osr);
|
|
}
|
|
}
|
|
api_region *sr = srw->getSubtractorRegion();
|
|
if (sr != NULL && !sr->isEmpty())
|
|
{
|
|
api_region *osr = sr->clone();
|
|
RECT r;
|
|
srw->getNonClientRect(&r);
|
|
r.left -= gr.left;
|
|
r.top -= gr.top;
|
|
osr->offset(r.left, r.top);
|
|
subtractorrgn->addRegion(osr);
|
|
sr->disposeClone(osr);
|
|
}
|
|
}
|
|
|
|
if (andme)
|
|
{
|
|
wr->andRegion(andme);
|
|
delete andme;
|
|
}
|
|
if (orme)
|
|
{
|
|
wr->addRegion(orme);
|
|
delete orme;
|
|
}
|
|
if (subme)
|
|
{
|
|
wr->subtractRgn(subme);
|
|
delete subme;
|
|
}
|
|
|
|
composedrgn->addRegion(wr);
|
|
reg->disposeClone(wr);
|
|
delete _reg;
|
|
}
|
|
|
|
void BaseWnd::updateWindowRegion()
|
|
{
|
|
if (!isPostOnInit() || isVirtual()) return ;
|
|
if (getDesktopAlpha())
|
|
{
|
|
// if desktopalpha is on, we can't use regions (thanks MS), we have to rely on the framebuffer correctness
|
|
//CUT SetWindowRgn(getOsWindowHandle(), NULL, FALSE);
|
|
Wasabi::Std::Wnd::setWndRegion(getOsWindowHandle(), NULL);
|
|
return ;
|
|
}
|
|
api_region *_r = getComposedRegion();
|
|
api_region *_s = getSubtractorRegion();
|
|
ASSERT(_r != NULL && _s != NULL);
|
|
|
|
api_region *z = _r->clone();
|
|
z->subtractRgn(_s);
|
|
|
|
assignWindowRegion(z);
|
|
_r->disposeClone(z);
|
|
}
|
|
|
|
// wr is NOT scaled!!!
|
|
void BaseWnd::assignWindowRegion(api_region *wr)
|
|
{
|
|
ASSERT(wr != NULL);
|
|
|
|
if (!isPostOnInit()) return ;
|
|
|
|
int isrect = wr->isRect();
|
|
|
|
RECT r;
|
|
BaseWnd::getWindowRect(&r);
|
|
|
|
//DebugStringW( L"\nBaseWnd::assignWindowRegion() r before - x = %d, y = %d, w = %d, h = %d \n", r.left, r.top, r.right - r.left, r.bottom - r.top );
|
|
|
|
r.right -= r.left;
|
|
r.left = 0;
|
|
r.bottom -= r.top;
|
|
r.top = 0;
|
|
|
|
//DebugStringW( L"BaseWnd::assignWindowRegion() r after - x = %d, y = %d, w = %d, h = %d \n", r.left, r.top, r.right - r.left, r.bottom - r.top );
|
|
|
|
RECT z;
|
|
wr->getBox(&z);
|
|
|
|
//DebugStringW( L"BaseWnd::assignWindowRegion() z before - x = %d, y = %d, w = %d, h = %d \n", z.left, z.top, z.right - z.left, z.bottom - z.top );
|
|
|
|
z.left = 0;
|
|
z.top = 0;
|
|
|
|
//DebugStringW( L"BaseWnd::assignWindowRegion() z after - x = %d, y = %d, w = %d, h = %d \n", z.left, z.top, z.right - z.left, z.bottom - z.top );
|
|
|
|
if (renderRatioActive())
|
|
{
|
|
double i = getRenderRatio();
|
|
wr->scale(i, i, FALSE);
|
|
}
|
|
|
|
RECT sz;
|
|
wr->getBox(&sz);
|
|
|
|
//DebugStringW( L"BaseWnd::assignWindowRegion() sz before - x = %d, y = %d, w = %d, h = %d \n", sz.left, sz.top, sz.right - sz.left, sz.bottom - sz.top );
|
|
|
|
sz.right -= sz.left;
|
|
sz.bottom -= sz.top;
|
|
sz.left = 0;
|
|
sz.top = 0;
|
|
|
|
//DebugStringW( L"BaseWnd::assignWindowRegion() sz after - x = %d, y = %d, w = %d, h = %d \n", sz.left, sz.top, sz.right - sz.left, sz.bottom - sz.top );
|
|
|
|
if (isrect &&
|
|
((z.right == rwidth && z.bottom == rheight) ||
|
|
(sz.left == r.left && sz.right == r.right && sz.top == r.top && sz.bottom == r.bottom) ||
|
|
(0)
|
|
)
|
|
)
|
|
{
|
|
//CUT SetWindowRgn(getOsWindowHandle(), NULL, TRUE);
|
|
if (!lastnullregion)
|
|
{
|
|
Wasabi::Std::Wnd::setWndRegion(getOsWindowHandle(), NULL, TRUE);
|
|
lastnullregion = 1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//DebugStringW(L"setting region, rwidth = %d, rheight = %d, z.right = %d, z.bottom = %d\n", rwidth, rheight, z.right, z.bottom);
|
|
//CUT SetWindowRgn(getOsWindowHandle(), wr->makeWindowRegion(), TRUE);
|
|
Wasabi::Std::Wnd::setWndRegion(getOsWindowHandle(), wr->makeWindowRegion(), TRUE);
|
|
lastnullregion = 0;
|
|
}
|
|
}
|
|
|
|
void BaseWnd::performBatchProcesses()
|
|
{
|
|
// recompute the window region if needed and apply it to the HWND
|
|
if (wndregioninvalid && !isVirtual())
|
|
if (ensureWindowRegionValid())
|
|
updateWindowRegion();
|
|
if (need_flush_cascaderepaint)
|
|
{
|
|
_cascadeRepaintRgn(deferedCascadeRepaintRgn);
|
|
deferedCascadeRepaintRgn->empty();
|
|
need_flush_cascaderepaint = 0;
|
|
}
|
|
}
|
|
|
|
int BaseWnd::getRegionOp()
|
|
{
|
|
return regionop;
|
|
}
|
|
|
|
void BaseWnd::setRegionOp(int op)
|
|
{
|
|
if (regionop != op)
|
|
{
|
|
regionop = op;
|
|
invalidateWindowRegion();
|
|
}
|
|
}
|
|
|
|
int BaseWnd::isRectRgn()
|
|
{
|
|
return rectrgn;
|
|
}
|
|
|
|
void BaseWnd::setRectRgn(int i)
|
|
{
|
|
rectrgn = i;
|
|
}
|
|
|
|
TimerClient *BaseWnd::timerclient_getMasterClient()
|
|
{
|
|
if (!isVirtual()) return this;
|
|
ifc_window *w = getParent();
|
|
if (w)
|
|
{
|
|
TimerClient *tc = w->getTimerClient();
|
|
if (tc)
|
|
return tc->timerclient_getMasterClient();
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void BaseWnd::timerclient_onMasterClientMultiplex()
|
|
{
|
|
performBatchProcesses();
|
|
}
|
|
|
|
TimerClient *BaseWnd::getTimerClient()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
ifc_dependent *BaseWnd::rootwnd_getDependencyPtr()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
ifc_dependent *BaseWnd::timerclient_getDependencyPtr()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
void BaseWnd::addMinMaxEnforcer(ifc_window *w)
|
|
{
|
|
minmaxEnforcers.addItem(w);
|
|
signalMinMaxEnforcerChanged();
|
|
}
|
|
|
|
void BaseWnd::removeMinMaxEnforcer(ifc_window *w)
|
|
{
|
|
minmaxEnforcers.removeItem(w);
|
|
signalMinMaxEnforcerChanged();
|
|
}
|
|
|
|
void BaseWnd::signalMinMaxEnforcerChanged(void)
|
|
{
|
|
ifc_window *w = getDesktopParent();
|
|
if (w == NULL || w == this) onMinMaxEnforcerChanged();
|
|
else w->signalMinMaxEnforcerChanged();
|
|
}
|
|
|
|
int BaseWnd::getNumMinMaxEnforcers()
|
|
{
|
|
return minmaxEnforcers.getNumItems();
|
|
}
|
|
|
|
ifc_window *BaseWnd::enumMinMaxEnforcer(int n)
|
|
{
|
|
return minmaxEnforcers.enumItem(n);
|
|
}
|
|
|
|
int BaseWnd::onAction(const wchar_t *action, const wchar_t *param, int x, int y, intptr_t p1, intptr_t p2, void *data, size_t datalen, ifc_window *source)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::sendAction(ifc_window *target, const wchar_t *action, const wchar_t *param, int x, int y, intptr_t p1, intptr_t p2, void *data, size_t datalen)
|
|
{
|
|
ASSERT(target != NULL);
|
|
return target->onAction(action, param, x, y, p1, p2, data, datalen, this);
|
|
}
|
|
|
|
int BaseWnd::virtualBeforePaint(api_region *r)
|
|
{
|
|
if (!virtualCanvas) return 0;
|
|
PaintCallbackInfoI pc(virtualCanvas, r);
|
|
dependent_sendEvent(BaseWnd::depend_getClassGuid(), Event_ONPAINT, PaintCallback::BEFOREPAINT, &pc);
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::virtualAfterPaint(api_region *r)
|
|
{
|
|
if (!virtualCanvas) return 0;
|
|
PaintCallbackInfoI pc(virtualCanvas, r);
|
|
dependent_sendEvent(BaseWnd::depend_getClassGuid(), Event_ONPAINT, PaintCallback::AFTERPAINT, &pc);
|
|
return 1;
|
|
}
|
|
|
|
int BaseWnd::timerclient_onDeferredCallback(intptr_t p1, intptr_t p2)
|
|
{
|
|
TimerClientI::timerclient_onDeferredCallback(p1, p2);
|
|
return onDeferredCallback(p1, p2);
|
|
}
|
|
|
|
void BaseWnd::timerclient_timerCallback(int id)
|
|
{
|
|
TimerClientI::timerclient_timerCallback(id);
|
|
timerCallback(id);
|
|
}
|
|
|
|
int BaseWnd::setTimer(int id, int ms)
|
|
{
|
|
return timerclient_setTimer(id, ms);
|
|
}
|
|
|
|
int BaseWnd::killTimer(int id)
|
|
{
|
|
return timerclient_killTimer(id);
|
|
}
|
|
|
|
void BaseWnd::postDeferredCallback(intptr_t p1, intptr_t p2, int mindelay)
|
|
{
|
|
#ifdef _WIN32
|
|
// TODO: re-enable, but post to some other window (e.g. some singleton window), not this window
|
|
// because our message pump might be blocked
|
|
// maybe make a hidden window in api_timer for this purpose
|
|
|
|
//if (mindelay)
|
|
timerclient_postDeferredCallback(p1, p2, mindelay);
|
|
//else
|
|
//PostMessage(hwnd, WM_DEFER_CALLBACK, p1, p2);
|
|
#else
|
|
#warning "port me - I can be optimized - don't use timers for this, use mac os x equiv of PostMessage!"
|
|
timerclient_postDeferredCallback(p1, p2, mindelay);
|
|
#endif
|
|
}
|
|
|
|
int BaseWnd::bypassModal()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::setRenderBaseTexture(int r)
|
|
{
|
|
renderbasetexture = r;
|
|
if (isInited()) invalidate();
|
|
}
|
|
|
|
int BaseWnd::getRenderBaseTexture()
|
|
{
|
|
return renderbasetexture;
|
|
}
|
|
|
|
GuiObject *BaseWnd::getGuiObject()
|
|
{
|
|
if (my_guiobject == NULL)
|
|
{
|
|
my_guiobject = static_cast<GuiObject *>(getInterface(guiObjectGuid));
|
|
}
|
|
return my_guiobject;
|
|
}
|
|
|
|
//CUT someday
|
|
int BaseWnd::getFlag(int flag)
|
|
{
|
|
/* switch (flag) {
|
|
}*/
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::triggerEvent(int event, intptr_t p1, intptr_t p2)
|
|
{
|
|
//PORTME
|
|
switch (event)
|
|
{
|
|
case TRIGGER_ONRESIZE:
|
|
if (isPostOnInit())
|
|
onResize();
|
|
break;
|
|
case TRIGGER_INVALIDATE:
|
|
if (isPostOnInit())
|
|
invalidate();
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void BaseWnd::registerAcceleratorSection(const wchar_t *name, int global)
|
|
{
|
|
#if defined(WASABI_COMPILE_LOCALES)
|
|
WASABI_API_LOCALE->locales_registerAcceleratorSection(name, this, global);
|
|
#endif
|
|
}
|
|
|
|
int BaseWnd::onAcceleratorEvent(const wchar_t *name)
|
|
{
|
|
for (int i = 0;i < getNumRootWndChildren();i++)
|
|
if (enumRootWndChildren(i)->onAcceleratorEvent(name))
|
|
return 1;
|
|
return 0;
|
|
}
|
|
|
|
int BaseWnd::allowDeactivation()
|
|
{
|
|
return allow_deactivate & ((WASABI_API_WND == NULL) || WASABI_API_WND->appdeactivation_isallowed(this));
|
|
}
|
|
|
|
void BaseWnd::onMinimize()
|
|
{
|
|
if (!isVirtual())
|
|
{
|
|
dropVirtualCanvas();
|
|
}
|
|
}
|
|
|
|
void BaseWnd::dropVirtualCanvas()
|
|
{
|
|
deferedInvalidate();
|
|
delete virtualCanvas;
|
|
virtualCanvas = NULL;
|
|
}
|
|
|
|
void BaseWnd::onRestore()
|
|
{
|
|
if (getDesktopParent() == this)
|
|
{
|
|
cascadeRepaint(TRUE);
|
|
}
|
|
}
|
|
|
|
ifc_window *BaseWnd::findWindow(const wchar_t *id)
|
|
{
|
|
RootWndFinder find_object;
|
|
find_object.reset();
|
|
find_object.setFindId(id);
|
|
ifc_window *ret = findWindowChain(&find_object);
|
|
|
|
#ifdef _DEBUG
|
|
if (ret == NULL)
|
|
DebugStringW(L"findWindow : window not found by id ! %s \n", id);
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
ifc_window *BaseWnd::findWindowByInterface(GUID interface_guid)
|
|
{
|
|
RootWndFinder find_object;
|
|
find_object.reset();
|
|
find_object.setFindGuid(interface_guid);
|
|
ifc_window *ret = findWindowChain(&find_object);
|
|
|
|
#ifdef _DEBUG
|
|
char str[256] = {0};
|
|
nsGUID::toChar(interface_guid, str);
|
|
if (ret == NULL)
|
|
DebugStringW(L"findWindow : object not found by guid ! %s \n", str);
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
ifc_window *BaseWnd::findWindowByCallback(FindObjectCallback *cb)
|
|
{
|
|
ifc_window *ret = findWindowChain(cb);
|
|
#ifdef _DEBUG
|
|
if (ret == NULL)
|
|
DebugStringW(L"findWindow : object not found by callback!\n");
|
|
#endif
|
|
|
|
return ret;
|
|
}
|
|
|
|
ifc_window *BaseWnd::findWindowChain(FindObjectCallback *cb, ifc_window *wcaller)
|
|
{
|
|
|
|
if (!cb) return NULL;
|
|
|
|
if (cb->findobjectcb_matchObject(this)) return this;
|
|
|
|
// first lets not look in subdirectories
|
|
|
|
for (int i = 0;i < getNumRootWndChildren();i++)
|
|
{
|
|
ifc_window *child = enumRootWndChildren(i);
|
|
if (!child || child == wcaller) continue;
|
|
if (cb->findobjectcb_matchObject(child))
|
|
return child;
|
|
}
|
|
|
|
// ok so it wasn't in our content, lets try to find it as a grandchildren
|
|
|
|
for (int i = 0;i < getNumRootWndChildren();i++)
|
|
{
|
|
ifc_window *child = enumRootWndChildren(i);
|
|
if (child->getNumRootWndChildren() > 0)
|
|
{
|
|
ifc_window *ret = child->findWindowChain(cb, this);
|
|
if (ret) return ret;
|
|
}
|
|
}
|
|
|
|
// so it wasnt one of our children, we'll hop the tree up one level and ask our parent to find it
|
|
// for us. of course, our parents are smart, they won't ask us back when asking our sibblings
|
|
ifc_window *p = getParent();
|
|
if (p != NULL && wcaller != p)
|
|
{
|
|
return p->findWindowChain(cb, this);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
const wchar_t *BaseWnd::timerclient_getName()
|
|
{
|
|
tcname = StringPrintfW(L"name=\"%S\", id=\"%s\"", getRootWndName(), getId());
|
|
return tcname;
|
|
}
|
|
|
|
void BaseWnd::setTabOrder(int a)
|
|
{
|
|
if (getParent() != NULL)
|
|
getParent()->setVirtualTabOrder(this, a);
|
|
}
|
|
|
|
int BaseWnd::getTabOrder()
|
|
{
|
|
if (getParent() != NULL)
|
|
return getParent()->getVirtualTabOrder(this);
|
|
return -1;
|
|
}
|
|
|
|
void BaseWnd::recursive_setVirtualTabOrder(ifc_window *w, float a, float lambda)
|
|
{
|
|
ASSERT(w != NULL);
|
|
childtabs.setAutoSort(0);
|
|
int i = getTabOrderEntry(a);
|
|
if (i != -1)
|
|
{
|
|
TabOrderEntry *toe = childtabs.enumItem(i);
|
|
if (toe->wnd != w)
|
|
{
|
|
lambda += TABORDER_K;
|
|
if (lambda != 1.0)
|
|
recursive_setVirtualTabOrder(toe->wnd, a + lambda, lambda);
|
|
}
|
|
else
|
|
{
|
|
return ;
|
|
}
|
|
}
|
|
i = getTabOrderEntry(w);
|
|
if (i != -1)
|
|
{
|
|
delete childtabs.enumItem(i);
|
|
childtabs.removeByPos(i);
|
|
}
|
|
TabOrderEntry *toe = new TabOrderEntry;
|
|
toe->wnd = w;
|
|
toe->order = a;
|
|
childtabs.addItem(toe);
|
|
}
|
|
|
|
void BaseWnd::setVirtualTabOrder(ifc_window *w, int a)
|
|
{
|
|
if (a == -1)
|
|
{
|
|
delTabOrderEntry(w);
|
|
return ;
|
|
}
|
|
recursive_setVirtualTabOrder(w, (float)a);
|
|
}
|
|
|
|
int BaseWnd::getVirtualTabOrder(ifc_window *w)
|
|
{
|
|
int a = (int)getTabOrderEntry(w);
|
|
if (a == -1) return -1;
|
|
return (int)childtabs.enumItem(a);
|
|
}
|
|
|
|
int BaseWnd::getTabOrderEntry(ifc_window *w)
|
|
{
|
|
foreach(childtabs)
|
|
if (childtabs.getfor()->wnd == w)
|
|
return foreach_index;
|
|
endfor;
|
|
return -1;
|
|
}
|
|
|
|
void BaseWnd::delTabOrderEntry(int i)
|
|
{
|
|
int a = getTabOrderEntry((float)i);
|
|
if (a == -1) return ;
|
|
childtabs.removeByPos(a);
|
|
}
|
|
|
|
void BaseWnd::delTabOrderEntry(ifc_window *w)
|
|
{
|
|
int a = getTabOrderEntry(w);
|
|
if (a == -1) return ;
|
|
delete childtabs.enumItem(a);
|
|
childtabs.removeByPos(a);
|
|
}
|
|
|
|
int BaseWnd::getTabOrderEntry(float order)
|
|
{
|
|
foreach(childtabs)
|
|
if (childtabs.getfor()->order == order)
|
|
return foreach_index;
|
|
endfor;
|
|
return -1;
|
|
}
|
|
|
|
void BaseWnd::setAutoTabOrder()
|
|
{
|
|
if (!getParent()) return ;
|
|
getParent()->setVirtualAutoTabOrder(this);
|
|
}
|
|
|
|
void BaseWnd::setVirtualAutoTabOrder(ifc_window *w)
|
|
{
|
|
delTabOrderEntry(w);
|
|
float o = 0;
|
|
for (int i = 0;i < childtabs.getNumItems();i++)
|
|
{
|
|
o = MAX(o, childtabs.enumItem(i)->order);
|
|
}
|
|
setVirtualTabOrder(w, ((int)o) + 1);
|
|
}
|
|
|
|
void BaseWnd::focusNext()
|
|
{
|
|
ifc_window *dp = getDesktopParent();
|
|
if (dp != this)
|
|
{
|
|
if (dp != NULL)
|
|
dp->focusNext();
|
|
return ;
|
|
}
|
|
ifc_window *w = getTab(TAB_GETNEXT);
|
|
if (w != NULL) w->setFocus();
|
|
}
|
|
|
|
void BaseWnd::focusPrevious()
|
|
{
|
|
ifc_window *dp = getDesktopParent();
|
|
if (dp != this)
|
|
{
|
|
if (dp != NULL)
|
|
getDesktopParent()->focusPrevious();
|
|
return ;
|
|
}
|
|
ifc_window *w = getTab(TAB_GETPREVIOUS);
|
|
if (w != NULL) w->setFocus();
|
|
}
|
|
|
|
void BaseWnd::recursive_buildTabList(ifc_window *from, PtrList<ifc_window> *list)
|
|
{
|
|
for (int i = 0;i < from->getNumTabs();i++)
|
|
{
|
|
ifc_window *r = from->enumTab(i);
|
|
if (r->isVisible() && r->getPaintingAlpha() > 0)
|
|
{
|
|
if (r->wantFocus())
|
|
list->addItem(r);
|
|
recursive_buildTabList(r, list);
|
|
}
|
|
}
|
|
}
|
|
|
|
ifc_window *BaseWnd::getTab(int what)
|
|
{
|
|
PtrList<ifc_window> listnow;
|
|
|
|
recursive_buildTabList(this, &listnow);
|
|
|
|
int p = listnow.searchItem(rootfocus);
|
|
|
|
if (p == -1)
|
|
for (int i = 0; i < listnow.getNumItems(); i++)
|
|
{
|
|
ifc_window *r = listnow.enumItem(i);
|
|
if (r->gotFocus())
|
|
{
|
|
//DebugString("desync of rootfocus, fixing\n");
|
|
p = i;
|
|
assignRootFocus(r);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (what == TAB_GETNEXT && rootfocus != NULL)
|
|
{
|
|
|
|
p++;
|
|
if (p >= listnow.getNumItems())
|
|
p = 0;
|
|
return listnow.enumItem(p);
|
|
|
|
}
|
|
else if (what == TAB_GETPREVIOUS && rootfocus != NULL)
|
|
{
|
|
|
|
p--;
|
|
if (p < 0) p = listnow.getNumItems() - 1;
|
|
return listnow.enumItem(p);
|
|
|
|
}
|
|
else if (what == TAB_GETCURRENT)
|
|
{
|
|
|
|
return rootfocus;
|
|
|
|
}
|
|
else if (what == TAB_GETFIRST || (what == TAB_GETNEXT && rootfocus == NULL))
|
|
{
|
|
|
|
return listnow.getFirst();
|
|
|
|
}
|
|
else if (what == TAB_GETLAST || (what == TAB_GETPREVIOUS && rootfocus == NULL))
|
|
{
|
|
|
|
return listnow.getLast();
|
|
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int BaseWnd::getNumTabs()
|
|
{
|
|
return childtabs.getNumItems();
|
|
}
|
|
|
|
ifc_window *BaseWnd::enumTab(int i)
|
|
{
|
|
childtabs.sort();
|
|
return childtabs.enumItem(i)->wnd;
|
|
}
|
|
|
|
void BaseWnd::onSetRootFocus(ifc_window *w)
|
|
{
|
|
assignRootFocus(w);
|
|
ifc_window *dp = getDesktopParent();
|
|
if (dp && dp != this) dp->onSetRootFocus(w);
|
|
}
|
|
|
|
void BaseWnd::autoFocus(ifc_window *w)
|
|
{
|
|
if (w->getFocusOnClick() && w->wantFocus())
|
|
{
|
|
w->setFocus();
|
|
return ;
|
|
}
|
|
ifc_window *g = w;
|
|
while (1)
|
|
{
|
|
ifc_window *p = g->getParent();
|
|
if (p == NULL) break;
|
|
ifc_window *dp = p->getDesktopParent();
|
|
if (dp && dp != p)
|
|
{
|
|
if (p->wantFocus() && p->getFocusOnClick())
|
|
{
|
|
p->setFocus();
|
|
return ;
|
|
}
|
|
g = p;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
}
|
|
|
|
void BaseWnd::setNoLeftClicks(int no)
|
|
{
|
|
noleftclick = no;
|
|
}
|
|
|
|
void BaseWnd::setNoRightClicks(int no)
|
|
{
|
|
norightclick = no;
|
|
}
|
|
|
|
void BaseWnd::setNoDoubleClicks(int no)
|
|
{
|
|
nodoubleclick = no;
|
|
}
|
|
|
|
void BaseWnd::setNoMouseMoves(int no)
|
|
{
|
|
nomousemove = no;
|
|
}
|
|
|
|
void BaseWnd::setNoContextMenus(int no)
|
|
{
|
|
nocontextmnu = no;
|
|
}
|
|
|
|
void BaseWnd::setDefaultCursor(Cursor *c)
|
|
{
|
|
customdefaultcursor = c;
|
|
}
|
|
|
|
|
|
OSCURSORHANDLE BaseWnd::getCustomCursor(int x, int y)
|
|
{
|
|
#ifdef _WIN32
|
|
return customdefaultcursor ? customdefaultcursor->getOSHandle() : NULL;
|
|
#else
|
|
#warning port me
|
|
return 0;
|
|
#endif
|
|
}
|
|
|
|
Accessible *BaseWnd::createNewAccObj()
|
|
{
|
|
waServiceFactory *f = WASABI_API_SVC->service_enumService(WaSvc::ACCESSIBILITY, 0);
|
|
if (f != NULL)
|
|
{
|
|
svc_accessibility *svc = castService<svc_accessibility>(f);
|
|
if (svc != NULL)
|
|
{
|
|
Accessible *a = svc->createAccessibleObject(this);
|
|
WASABI_API_SVC->service_release(svc);
|
|
return a;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
Accessible *BaseWnd::getAccessibleObject(int createifnotexist)
|
|
{
|
|
if (!createifnotexist) return accessible;
|
|
if (!accessible)
|
|
accessible = createNewAccObj();
|
|
else
|
|
accessible->addRef();
|
|
return accessible;
|
|
}
|
|
|
|
int BaseWnd::accessibility_getState()
|
|
{
|
|
int state = 0;
|
|
if (!isVisible()) state |= STATE_SYSTEM_INVISIBLE;
|
|
//if (isVirtual() && !wantFocus()) state |= STATE_SYSTEM_INVISIBLE;
|
|
if (gotFocus()) state |= STATE_SYSTEM_FOCUSED;
|
|
return state;
|
|
}
|
|
|
|
void BaseWnd::activate()
|
|
{
|
|
Wasabi::Std::Wnd::setActiveWindow(getRootParent()->getOsWindowHandle());
|
|
}
|
|
|
|
void BaseWnd::setOSWndName(const wchar_t *name)
|
|
{
|
|
if (isVirtual()) return ;
|
|
//#ifdef COMPILE_WASABI_SKIN // for some reason this isn't being correctly defined
|
|
if (name)
|
|
{
|
|
Wasabi::Std::Wnd::setWndName(getOsWindowHandle(), name);
|
|
}
|
|
else
|
|
Wasabi::Std::Wnd::setWndName(getOsWindowHandle(), L"");
|
|
|
|
}
|
|
|
|
const wchar_t *BaseWnd::getOSWndName()
|
|
{
|
|
if (isVirtual()) return NULL;
|
|
wchar_t str[4096] = {0};
|
|
Wasabi::Std::Wnd::getWndName(getOsWindowHandle(), str, 4095);
|
|
str[4095] = '\0';
|
|
osname = str;
|
|
return osname;
|
|
}
|
|
|
|
#ifdef EXPERIMENTAL_INDEPENDENT_AOT
|
|
void BaseWnd::setAlwaysOnTop(int i)
|
|
{
|
|
// this function should not optimize itself
|
|
if (getDesktopParent() == this)
|
|
{
|
|
if (i)
|
|
{
|
|
//CUT SetWindowPos(getOsWindowHandle(), HWND_TOPMOST, 0,0,0,0, SWP_NOMOVE|SWP_NOSIZE|SWP_NOOWNERZORDER);
|
|
Wasabi::Std::Wnd::setTopmost(getOsWindowHandle(), TRUE);
|
|
}
|
|
else
|
|
{
|
|
saveTopMosts();
|
|
//CUT SetWindowPos(getOsWindowHandle(), HWND_NOTOPMOST, 0,0,0,0, SWP_NOMOVE|SWP_NOSIZE|SWP_NOOWNERZORDER);
|
|
Wasabi::Std::Wnd::setTopmost(getOsWindowHandle(), FALSE);
|
|
restoreTopMosts();
|
|
}
|
|
alwaysontop = i;
|
|
return ;
|
|
}
|
|
ifc_window *p = getParent();
|
|
if (p != NULL)
|
|
p->setAlwaysOnTop(i);
|
|
}
|
|
|
|
int BaseWnd::getAlwaysOnTop()
|
|
{
|
|
if (getDesktopParent() == this)
|
|
return alwaysontop;
|
|
ifc_window *p = getParent();
|
|
if (!p) return 0;
|
|
return p->getAlwaysOnTop();
|
|
}
|
|
#endif
|
|
|
|
void BaseWnd::wndwatcher_onDeleteWindow(ifc_window *w)
|
|
{
|
|
if (w == rootfocus)
|
|
{
|
|
rootfocus = NULL;
|
|
}
|
|
}
|
|
|
|
void BaseWnd::assignRootFocus(ifc_window *w)
|
|
{
|
|
rootfocuswatcher.watchWindow(w);
|
|
rootfocus = w;
|
|
}
|
|
|
|
|
|
Canvas *BaseWnd::getFrameBuffer()
|
|
{
|
|
return virtualCanvas;
|
|
}
|
|
|
|
void BaseWnd::setForeignWnd(int i)
|
|
{
|
|
m_takenOver = i;
|
|
}
|
|
|
|
int BaseWnd::bufferizeLockedUIMsg(int uMsg, int wParam, int lParam)
|
|
{
|
|
if (WASABI_API_SKIN && !WASABI_API_SKIN->skin_getLockUI()) return 0;
|
|
if (!uiwaslocked)
|
|
{
|
|
uiwaslocked = 1;
|
|
setTimer(BUFFEREDMSG_TIMER_ID, 20);
|
|
}
|
|
bufferedMsgStruct msg;
|
|
msg.msg = uMsg;
|
|
msg.wparam = wParam;
|
|
msg.lparam = lParam;
|
|
bufferedmsgs.addItem(msg);
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::checkLockedUI()
|
|
{
|
|
//PORTME :(
|
|
#ifdef _WIN32
|
|
|
|
if (WASABI_API_SKIN && !WASABI_API_SKIN->skin_getLockUI())
|
|
{
|
|
uiwaslocked = 0;
|
|
killTimer(BUFFEREDMSG_TIMER_ID);
|
|
while (bufferedmsgs.getNumItems() > 0)
|
|
{
|
|
bufferedMsgStruct msg = bufferedmsgs.enumItem(0);
|
|
bufferedmsgs.delByPos(0);
|
|
SendMessageW(gethWnd(), msg.msg, msg.wparam, msg.lparam);
|
|
}
|
|
uiwaslocked = 0;
|
|
killTimer(BUFFEREDMSG_TIMER_ID);
|
|
}
|
|
#else
|
|
#warning port me
|
|
#endif
|
|
}
|
|
|
|
int BaseWnd::isMinimized()
|
|
{
|
|
ifc_window *w = getDesktopParent();
|
|
if (w == this || w == NULL) return minimized;
|
|
return w->isMinimized();
|
|
}
|
|
|
|
int BaseWnd::reinit()
|
|
{
|
|
#ifdef _WIN32
|
|
int nochild = (GetWindowLong(gethWnd(), GWL_STYLE) & WS_POPUP) ? 1 : 0;
|
|
int r = reinit(parentWnd ? parentWnd : WASABI_API_WND->main_getRootWnd(), nochild);
|
|
|
|
if (w2k_alpha)
|
|
setLayeredWindow(1);
|
|
|
|
return r;
|
|
#else
|
|
#warning port me!
|
|
#endif
|
|
}
|
|
|
|
|
|
int BaseWnd::reinit(ifc_window *parWnd, int nochild)
|
|
{
|
|
OSWINDOWHANDLE phwnd = parWnd->getOsWindowHandle();
|
|
ASSERT(phwnd != NULL);
|
|
int ret;
|
|
if (!nochild) parentWnd = parWnd;
|
|
else parentWnd = NULL;
|
|
ret = reinit(parWnd->getOsModuleHandle(), phwnd, nochild);
|
|
if (!ret) parentWnd = NULL; // abort
|
|
return ret;
|
|
}
|
|
|
|
int BaseWnd::reinit(OSMODULEHANDLE moduleHandle, OSWINDOWHANDLE parent, int nochild)
|
|
{
|
|
RECT r;
|
|
int w, h;
|
|
|
|
onBeforeReinit();
|
|
|
|
pushWindowRect();
|
|
preventcancelcapture = 1;
|
|
|
|
int _isvisible = isVisible(1);
|
|
int hadcapture = inputCaptured;
|
|
//DebugString("had capture = %d\n", hadcapture);
|
|
Wasabi::Std::Wnd::releaseCapture();
|
|
|
|
unparentHWNDChildren();
|
|
|
|
BaseWnd::getClientRect(&r);
|
|
|
|
hinstance = moduleHandle;
|
|
|
|
ASSERT(hinstance != NULL);
|
|
|
|
w = (r.right - r.left);
|
|
h = (r.bottom - r.top);
|
|
|
|
rwidth = w;
|
|
rheight = h;
|
|
rx = r.left;
|
|
ry = r.top;
|
|
|
|
WASABI_API_WND->appdeactivation_push_disallow(this);
|
|
|
|
// destroy old window
|
|
Wasabi::Std::Wnd::hideWnd(hwnd); //Wasabi::Std::Wnd::destroyWnd(hwnd);
|
|
ghosthwnd.addItem(hwnd);
|
|
|
|
hwnd = Wasabi::Std::Wnd::createWnd(&r, nochild, acceptExternalDrops(), parent, hinstance, static_cast<ifc_window*>(this));
|
|
#ifdef __APPLE__
|
|
#warning remove me
|
|
Wasabi::Std::Wnd::showWnd(hwnd);
|
|
#endif
|
|
|
|
WASABI_API_WND->appdeactivation_pop_disallow(this);
|
|
|
|
//ASSERT(hwnd != NULL); // lets fail nicely, this could happen for some win32 reason, we don't want to fail the whole app for it, so lets just fail the wnd
|
|
if (hwnd == NULL)
|
|
{
|
|
preventcancelcapture = 0;
|
|
return 0;
|
|
}
|
|
|
|
//CUT nreal++;
|
|
|
|
#ifdef _WIN32
|
|
//FUCKO
|
|
#ifdef URLDROPS
|
|
if (acceptExternalDrops()) RegisterDragDrop(hwnd, &m_target);
|
|
#elif !defined(WA3COMPATIBILITY)
|
|
if (!m_target && WASABI_API_WND != NULL)
|
|
m_target = WASABI_API_WND->getDefaultDropTarget();
|
|
if (m_target != NULL)
|
|
{
|
|
RegisterDragDrop(hwnd, (IDropTarget *)m_target);
|
|
}
|
|
#endif
|
|
#else
|
|
#warning port me - register drag & drop
|
|
#endif
|
|
|
|
this_visible = _isvisible;
|
|
|
|
//onInit();
|
|
//this_visible = !start_hidden;
|
|
|
|
reparentHWNDChildren();
|
|
|
|
popWindowRect();
|
|
|
|
invalidateWindowRegion();
|
|
updateWindowRegion();
|
|
|
|
if (this_visible)
|
|
Wasabi::Std::Wnd::showWnd(hwnd, FALSE);
|
|
|
|
if (hadcapture)
|
|
{
|
|
Wasabi::Std::Wnd::setCapture(hwnd);
|
|
}
|
|
preventcancelcapture = 0;
|
|
|
|
forcedOnResize();
|
|
redrawHWNDChildren();
|
|
//onPostOnInit();
|
|
|
|
onAfterReinit();
|
|
|
|
#ifdef WASABI_ON_REPARENT
|
|
WASABI_ON_REINIT(getOsWindowHandle());
|
|
#endif
|
|
|
|
return 1;
|
|
}
|
|
|
|
ReparentWndEntry::ReparentWndEntry(OSWINDOWHANDLE _wnd, OSWINDOWHANDLE parentwnd)
|
|
{
|
|
wnd = _wnd;
|
|
Wasabi::Std::Wnd::getWindowRect(wnd, &rect);
|
|
Wasabi::Std::Wnd::screenToClient(wnd, (int *)&(rect.left), (int *)&(rect.top));
|
|
Wasabi::Std::Wnd::clientToScreen(parentwnd, (int *)&(rect.left), (int *)&(rect.top));
|
|
}
|
|
|
|
void ReparentWndEntry::unparent()
|
|
{
|
|
Wasabi::Std::Wnd::setWndPos(wnd, NULL, rect.left, -30000, 0, 0, TRUE, TRUE, FALSE, FALSE, TRUE);
|
|
Wasabi::Std::Wnd::setParent(wnd, NULL);
|
|
}
|
|
|
|
void ReparentWndEntry::reparent(OSWINDOWHANDLE newparent)
|
|
{
|
|
Wasabi::Std::Wnd::setParent(wnd, newparent);
|
|
Wasabi::Std::Wnd::setWndPos(wnd, NULL, rect.left, rect.top, 0, 0, TRUE, TRUE, FALSE, FALSE, TRUE);
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
void BaseWnd::unparentHWNDChildren()
|
|
{
|
|
// just in case
|
|
reparentwnds.deleteAll();
|
|
|
|
#ifndef WIN32
|
|
#error port me ! // make a list of all the children oswindows and reparent them to the desktop somewhere we can't see
|
|
#endif
|
|
|
|
OSWINDOWHANDLE wnd = GetWindow(getOsWindowHandle(), GW_CHILD);
|
|
while (wnd)
|
|
{
|
|
reparentwnds.addItem(new ReparentWndEntry(wnd, getOsWindowHandle()));
|
|
wnd = GetWindow(wnd, GW_HWNDNEXT);
|
|
}
|
|
foreach(reparentwnds)
|
|
reparentwnds.getfor()->unparent();
|
|
endfor;
|
|
}
|
|
#endif
|
|
|
|
void BaseWnd::reparentHWNDChildren()
|
|
{
|
|
// reparent to the new oswindowhandle
|
|
foreach(reparentwnds)
|
|
reparentwnds.getfor()->reparent(getOsWindowHandle());
|
|
endfor;
|
|
}
|
|
|
|
void BaseWnd::redrawHWNDChildren()
|
|
{
|
|
// reparent to the new oswindowhandle
|
|
foreach(reparentwnds)
|
|
Wasabi::Std::Wnd::update(getOsWindowHandle());
|
|
endfor;
|
|
}
|
|
|
|
void BaseWnd::maximize(int axis)
|
|
{
|
|
//DebugString("maximize!\n");
|
|
// if already maximized, don't use current rect, use restore_rect
|
|
if (!maximized)
|
|
{
|
|
restore_rect.left = rx;
|
|
restore_rect.top = ry;
|
|
restore_rect.right = rx + rwidth;
|
|
restore_rect.bottom = ry + rheight;
|
|
}
|
|
|
|
RECT nr = restore_rect;
|
|
RECT dr;
|
|
|
|
Wasabi::Std::getViewport(&dr, NULL, NULL, getOsWindowHandle(), 0);
|
|
|
|
if (axis & MAXIMIZE_WIDTH)
|
|
{
|
|
nr.left = dr.left;
|
|
nr.right = dr.right;
|
|
}
|
|
if (axis & MAXIMIZE_HEIGHT)
|
|
{
|
|
nr.top = dr.top;
|
|
nr.bottom = dr.bottom;
|
|
}
|
|
maximized = 1;
|
|
if (axis != 0) resize(&nr);
|
|
onMaximize();
|
|
}
|
|
|
|
void BaseWnd::restore(int what)
|
|
{
|
|
if (maximized)
|
|
{
|
|
//DebugString("restore!\n");
|
|
if (what == (RESTORE_X | RESTORE_Y | RESTORE_WIDTH | RESTORE_HEIGHT))
|
|
resize(&restore_rect);
|
|
else
|
|
{
|
|
resize((what & RESTORE_X) ? restore_rect.left : NOCHANGE,
|
|
(what & RESTORE_Y) ? restore_rect.top : NOCHANGE,
|
|
(what & RESTORE_WIDTH) ? restore_rect.right - restore_rect.left : NOCHANGE,
|
|
(what & RESTORE_HEIGHT) ? restore_rect.bottom - restore_rect.top : NOCHANGE);
|
|
}
|
|
maximized = 0;
|
|
onRestore();
|
|
}
|
|
}
|
|
|
|
void BaseWnd::pushWindowRect()
|
|
{
|
|
//DebugString("pushWindowRect\n");
|
|
RECT wr;
|
|
getWindowRect(&wr);
|
|
wr.right = wr.left + rwidth;
|
|
wr.bottom = wr.top + rheight;
|
|
windowrectstack.push(wr);
|
|
}
|
|
|
|
int BaseWnd::popWindowRect(RECT *rc, int applyhow)
|
|
{
|
|
//DebugString("popWindowRect\n");
|
|
if (windowrectstack.peek() == 0) return 0;
|
|
RECT _rc;
|
|
windowrectstack.pop(&_rc);
|
|
RECT r;
|
|
getWindowRect(&r);
|
|
divRatio(&r);
|
|
if (applyhow)
|
|
{
|
|
if (applyhow == PWR_POSITION)
|
|
{
|
|
move(_rc.left, _rc.top);
|
|
if (rc)
|
|
{
|
|
int w = r.right - r.left;
|
|
int h = r.bottom - r.top;
|
|
rc->left = _rc.left;
|
|
rc->top = _rc.top;
|
|
rc->right = rc->left + w;
|
|
rc->bottom = rc->top + h;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (applyhow & PWR_X) r.left = _rc.left;
|
|
if (applyhow & PWR_Y) r.top = _rc.top;
|
|
if (applyhow & PWR_WIDTH) r.right = r.left + (_rc.right - _rc.left);
|
|
if (applyhow & PWR_HEIGHT) r.bottom = r.top + (_rc.bottom - _rc.top);
|
|
resizeToRect(&r);
|
|
if (rc) *rc = _rc;
|
|
}
|
|
}
|
|
else if (rc) *rc = _rc;
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::setRestoredRect(RECT *r)
|
|
{
|
|
if (!r)
|
|
return ;
|
|
|
|
restore_rect = *r;
|
|
maximized = 1;
|
|
}
|
|
|
|
int BaseWnd::getRestoredRect(RECT *r)
|
|
{
|
|
if (!r)
|
|
return 0;
|
|
|
|
if (!maximized)
|
|
return 0;
|
|
|
|
*r = restore_rect;
|
|
|
|
return 1;
|
|
}
|
|
|
|
void BaseWnd::notifyDeferredMove(int x, int y)
|
|
{
|
|
rx = x;
|
|
ry = y;
|
|
}
|
|
|
|
void BaseWnd::setWindowTitle(const wchar_t *title)
|
|
{
|
|
Layout *l = static_cast<Layout *>(getInterface(layoutGuid));
|
|
if (l)
|
|
{
|
|
Container *c = l->getParentContainer();
|
|
if (c)
|
|
{
|
|
c->setName(title);
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef __APPLE__
|
|
OSStatus BaseWnd::eventHandler(EventHandlerCallRef inHandlerCallRef, EventRef inEvent, void *inUserData)
|
|
{
|
|
return eventNotHandledErr;
|
|
}
|
|
#endif
|