2718 lines
79 KiB
C++
2718 lines
79 KiB
C++
// DeskMovr.cpp : Implementation of CDeskMovr
|
|
#include "stdafx.h"
|
|
#pragma hdrstop
|
|
|
|
#include "deskmovr.h"
|
|
|
|
#define DEFAULT_INTERVAL 200 // check every 1/5th of a second.
|
|
#define DEFAULT_ENABLED TRUE
|
|
#define DETECT_TIMER_ID 2323
|
|
#ifndef SHDOC401_DLL
|
|
#define ANIMATE_TIMER_ID 2324
|
|
#define ANIMATE_TIMER_INTERVAL (60*1000)
|
|
#endif
|
|
|
|
#ifdef DEBUG
|
|
const static TCHAR sz_DM1[] = TEXT("no dragable part");
|
|
const static TCHAR sz_DM2[] = TEXT("caption menu button");
|
|
const static TCHAR sz_DM3[] = TEXT("caption close button");
|
|
const static TCHAR sz_DM4[] = TEXT("move the component");
|
|
const static TCHAR sz_DM5[] = TEXT("resize width and height from bottom right corner");
|
|
const static TCHAR sz_DM6[] = TEXT("resize width and height from top left corner");
|
|
const static TCHAR sz_DM7[] = TEXT("resize width and height from top right corner");
|
|
const static TCHAR sz_DM8[] = TEXT("resize width and height from bottom left corner");
|
|
const static TCHAR sz_DM9[] = TEXT("resize from the top edge");
|
|
const static TCHAR sz_DM10[] = TEXT("resize from the bottom edge");
|
|
const static TCHAR sz_DM11[] = TEXT("resize from the left edge");
|
|
const static TCHAR sz_DM12[] = TEXT("resize from the right edge");
|
|
|
|
const LPCTSTR g_szDragModeStr[] = {
|
|
sz_DM1,
|
|
sz_DM2,
|
|
sz_DM3,
|
|
sz_DM4,
|
|
sz_DM5,
|
|
sz_DM6,
|
|
sz_DM7,
|
|
sz_DM8,
|
|
sz_DM9,
|
|
sz_DM10,
|
|
sz_DM11,
|
|
sz_DM12
|
|
};
|
|
|
|
#endif // DEBUG
|
|
|
|
// Globals used to track cdeskmovr instances. Useful for optimizing the
|
|
// detection code so we can turn off the timer when the mouse is not over our
|
|
// window. We track the cdeskmovr instances only on the first thread that instantiates
|
|
// us to keep the code simple, this should be the active desktop case.
|
|
#define CDESKMOVR_TRACK_COUNT 16 // 2 is what we use now for the active desktop, but we need
|
|
// extra slots in the array due to the fact that a new instance
|
|
// is created before the old one is destroyed during refresh.
|
|
// Make the array large to handle nested refreshes!
|
|
HHOOK g_hMouseHook;
|
|
HHOOK g_hKeyboardHook;
|
|
DWORD g_dwHookThreadId;
|
|
#ifndef SHDOC401_DLL
|
|
BOOL g_fAnimTimer = FALSE;
|
|
#endif
|
|
typedef CDeskMovr *PDM;
|
|
PDM g_apDM[CDESKMOVR_TRACK_COUNT];
|
|
BOOL CombView_EnableAnimations(BOOL fEnable);
|
|
|
|
|
|
DWORD g_fIgnoreTimers = 0;
|
|
#define IGNORE_CONTEXTMENU_UP 0x0001
|
|
#define IGNORE_CAPTURE_SET 0x0002
|
|
|
|
#define GET_SKIP_COUNT (2 * ((GetDoubleClickTime() / m_lInterval) + 1))
|
|
|
|
MAKE_CONST_BSTR(s_sstrNameMember, L"name");
|
|
MAKE_CONST_BSTR(s_sstrHidden, L"hidden");
|
|
MAKE_CONST_BSTR(s_sstrVisible, L"visible");
|
|
MAKE_CONST_BSTR(s_sstrResizeableMember, L"resizeable");
|
|
|
|
// These were declared in shellprv.h, so use DEFINE instead of MAKE
|
|
DEFINE_CONST_BSTR(s_sstrIDMember, L"id");
|
|
DEFINE_CONST_BSTR(s_sstrSubSRCMember, L"subscribed_url");
|
|
DEFINE_CONST_BSTR(s_sstrSRCMember, L"src");
|
|
|
|
#define CAPTION_ONLY (m_ItemState & (IS_FULLSCREEN | IS_SPLIT))
|
|
#define ISNORMAL (m_ItemState & IS_NORMAL)
|
|
#define ISFULLSCREEN (m_ItemState & IS_FULLSCREEN)
|
|
#define ISSPLIT (m_ItemState & IS_SPLIT)
|
|
#define CAPTIONBAR_HOTAREA(cyDefaultCaption, cyCurrentCaption) (((cyCurrentCaption == 0) && CAPTION_ONLY) ? (cyDefaultCaption / 2) : 3 * cyDefaultCaption)
|
|
|
|
#define MAX_ID_LENGTH 5
|
|
|
|
void ObtainSavedStateForElem(IHTMLElement *pielem,
|
|
LPCOMPSTATEINFO pCompState, BOOL fRestoredState);
|
|
|
|
DWORD g_aDMtoCSPushed[] = { 0, CS_MENUPUSHED, CS_CLOSEPUSHED, CS_RESTOREPUSHED, CS_FULLSCREENPUSHED, CS_SPLITPUSHED };
|
|
DWORD g_aDMtoCSTracked[] = { 0, CS_MENUTRACKED, CS_CLOSETRACKED, CS_RESTORETRACKED, CS_FULLSCREENTRACKED, CS_SPLITTRACKED };
|
|
DWORD g_aDMDCfromDragMode[] = { 0, DMDC_MENU, DMDC_CLOSE, DMDC_RESTORE, DMDC_FULLSCREEN, DMDC_SPLIT };
|
|
#define PUSHED(dm) (g_aDMtoCSPushed[(dm)])
|
|
#define TRACKED(dm) (g_aDMtoCSTracked[(dm)])
|
|
#define DMDCFROMDM(dm) (g_aDMDCfromDragMode[(dm)])
|
|
|
|
// Trident will flash if you change the zindex, even if it's to the same index,
|
|
// so we prevent the no-op call.
|
|
HRESULT SafeZOrderSet(IHTMLStyle * pistyle, LONG lNewZIndex)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
VARIANT varZ;
|
|
|
|
ASSERT(pistyle);
|
|
pistyle->get_zIndex(&varZ);
|
|
|
|
// Does the component need to be moved to the top?
|
|
if ((VT_I4 != varZ.vt) || (varZ.lVal != lNewZIndex))
|
|
{
|
|
// Yes.
|
|
varZ.vt = VT_I4;
|
|
varZ.lVal = lNewZIndex;
|
|
hr = pistyle->put_zIndex(varZ);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
// Keyboard hook is installed when first instance of deskmovr is created. Used to implement the keyboard
|
|
// interface for accessing the deskmovr control. Hook is removed when there are no more deskmovr's
|
|
// being tracked.
|
|
LRESULT CALLBACK DeskMovr_KeyboardHook(int nCode, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LRESULT lRes;
|
|
BOOL fHaveMover = FALSE;
|
|
|
|
for (int i = 0; i < CDESKMOVR_TRACK_COUNT; i++) {
|
|
if (g_apDM[i])
|
|
{
|
|
g_apDM[i]->OnKeyboardHook(wParam, lParam);
|
|
fHaveMover = TRUE;
|
|
}
|
|
}
|
|
|
|
lRes = CallNextHookEx(g_hKeyboardHook, nCode, wParam, lParam);
|
|
|
|
if (!fHaveMover)
|
|
{
|
|
UnhookWindowsHookEx(g_hKeyboardHook);
|
|
g_hKeyboardHook = NULL;
|
|
}
|
|
|
|
return lRes;
|
|
}
|
|
|
|
// Helper function used to track cdeskmovr intances so that we can turn off the
|
|
// timer if the mouse leaves our window.
|
|
void TrackMover(PDM pdm, BOOL fAdd)
|
|
{
|
|
if (!g_dwHookThreadId)
|
|
g_dwHookThreadId = GetCurrentThreadId();
|
|
|
|
if (!g_hKeyboardHook && fAdd)
|
|
g_hKeyboardHook = SetWindowsHookEx(WH_KEYBOARD, DeskMovr_KeyboardHook, NULL, GetCurrentThreadId());
|
|
|
|
if (!fAdd || (g_dwHookThreadId == GetCurrentThreadId())) {
|
|
int i = 0;
|
|
PDM pdmFind = fAdd ? NULL : pdm;
|
|
PDM pdmAssign = fAdd ? pdm : NULL;
|
|
|
|
while (i < CDESKMOVR_TRACK_COUNT) {
|
|
if (g_apDM[i] == pdmFind) {
|
|
g_apDM[i] = pdmAssign;
|
|
break;
|
|
}
|
|
i++;
|
|
}
|
|
|
|
// If we ever fail to track a mover then we'll never be able to optimize
|
|
// again. Shouldn't happen in practice for the case we care about.
|
|
if (fAdd && (i >= CDESKMOVR_TRACK_COUNT))
|
|
g_dwHookThreadId = 0xffffffff;
|
|
|
|
ASSERT(!fAdd || (i < CDESKMOVR_TRACK_COUNT));
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
void AnimateToTray(HWND hwnd, LONG lLeft, LONG lTop, LONG lWidth, LONG lHeight)
|
|
{
|
|
HWND hwndTray;
|
|
if (hwndTray = FindWindow(c_szTrayClass, NULL))
|
|
{
|
|
RECT rcComp, rcTray;
|
|
|
|
SetRect(&rcComp, lLeft, lTop, lLeft + lWidth, lTop + lHeight);
|
|
MapWindowPoints(hwnd, NULL, (LPPOINT)&rcComp, 2);
|
|
|
|
GetWindowRect(hwndTray, &rcTray);
|
|
if ((rcTray.right - rcTray.left) > (rcTray.bottom - rcTray.top))
|
|
rcTray.left = rcTray.right - GetSystemMetrics(SM_CXSMICON);
|
|
else
|
|
rcTray.top = rcTray.bottom - GetSystemMetrics(SM_CYSMICON);
|
|
|
|
DrawAnimatedRects(hwnd, IDANI_CAPTION, (CONST RECT *)&rcComp, (CONST RECT *)&rcTray);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void AnimateComponent(HWND hwnd, LONG lLeftS, LONG lTopS, LONG lWidthS, LONG lHeightS,
|
|
LONG lLeftD, LONG lTopD, LONG lWidthD, LONG lHeightD)
|
|
{
|
|
RECT rcSource, rcDest;
|
|
|
|
SetRect(&rcSource, lLeftS, lTopS, lLeftS + lWidthS, lTopS + lHeightS);
|
|
SetRect(&rcDest, lLeftD, lTopD, lLeftD + lWidthD, lTopD + lHeightD);
|
|
|
|
// 98/10/02 vtan: Removed the mapping as not required.
|
|
|
|
// MapWindowPoints(hwnd, NULL, (LPPOINT)&rcSource, 2);
|
|
// MapWindowPoints(hwnd, NULL, (LPPOINT)&rcDest, 2);
|
|
|
|
DrawAnimatedRects(hwnd, IDANI_CAPTION, (CONST RECT *)&rcSource, (CONST RECT *)&rcDest);
|
|
}
|
|
|
|
// Hook is installed when we detect we can turn our tracking timer off. The first
|
|
// time we get a mouse event in the hook we reactivate all the movers and unhook
|
|
// ourself.
|
|
LRESULT CALLBACK DeskMovr_MouseHook(int nCode, WPARAM wParam, LPARAM lParam)
|
|
{
|
|
LRESULT lRes;
|
|
|
|
#ifndef SHDOC401_DLL
|
|
// If we are getting mouse messages then a portion of the window must be
|
|
// visible so enable animations.
|
|
CombView_EnableAnimations(TRUE);
|
|
#endif
|
|
|
|
for (int i = 0; i < CDESKMOVR_TRACK_COUNT; i++) {
|
|
if (g_apDM[i])
|
|
g_apDM[i]->SmartActivateMovr(ERROR_SUCCESS);
|
|
}
|
|
|
|
lRes = CallNextHookEx(g_hMouseHook, nCode, wParam, lParam);
|
|
|
|
UnhookWindowsHookEx(g_hMouseHook);
|
|
g_hMouseHook = NULL;
|
|
|
|
return lRes;
|
|
}
|
|
|
|
|
|
// CDeskMovr
|
|
|
|
CDeskMovr::CDeskMovr()
|
|
: m_TimerWnd(_T("STATIC"), this, 1)
|
|
{
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::CDeskMovr()");
|
|
|
|
m_fEnabled = DEFAULT_ENABLED;
|
|
m_lInterval = DEFAULT_INTERVAL;
|
|
|
|
m_cxSMBorder = GetSystemMetrics(SM_CXBORDER);
|
|
m_cySMBorder = GetSystemMetrics(SM_CYBORDER);
|
|
m_cxBorder = m_cxSMBorder;
|
|
m_cyBorder = m_cySMBorder;
|
|
m_cyCaption = 0;
|
|
|
|
m_dmCur = dmNull;
|
|
m_dmTrack = dmNull;
|
|
|
|
m_hcursor = LoadCursor(NULL, IDC_ARROW);
|
|
m_CaptionState = 0;
|
|
m_hwndParent;
|
|
|
|
m_fTimer = FALSE;
|
|
m_fCaptured = FALSE;
|
|
m_uiTimerID = DETECT_TIMER_ID;
|
|
m_pistyle = NULL;
|
|
|
|
m_pistyleTarget = NULL;
|
|
m_pielemTarget = NULL;
|
|
m_iSrcTarget = -1;
|
|
|
|
m_bstrTargetName = NULL;
|
|
|
|
m_dx = m_dy = 0;
|
|
|
|
m_top = m_left = m_width = m_height = 0;
|
|
|
|
// Tell ATL that we don't want to be Windowless
|
|
m_bWindowOnly = TRUE;
|
|
|
|
// Track this instance
|
|
TrackMover(this, TRUE);
|
|
}
|
|
|
|
CDeskMovr::~CDeskMovr(void)
|
|
{
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::~CDeskMovr() m_bstrTargetName=%ls.", GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
// clean up, detach from events, if necessary.
|
|
DeactivateMovr(TRUE);
|
|
|
|
if (m_bstrTargetName != NULL)
|
|
SysFreeString(m_bstrTargetName);
|
|
|
|
TrackMover(this, FALSE);
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::SmartActivateMovr(HRESULT hrPropagate)
|
|
{
|
|
if ((FALSE == m_nFreezeEvents) && m_fEnabled && !m_pielemTarget)
|
|
{
|
|
#ifndef SHDOC401_DLL
|
|
// Release our animation timer if it exists and create our regular one
|
|
if (g_fAnimTimer && (m_uiTimerID == ANIMATE_TIMER_ID))
|
|
{
|
|
m_TimerWnd.KillTimer(m_uiTimerID);
|
|
m_uiTimerID = DETECT_TIMER_ID;
|
|
g_fAnimTimer = FALSE;
|
|
m_fTimer = m_TimerWnd.SetTimer(m_uiTimerID, m_lInterval, NULL) != 0;
|
|
}
|
|
#endif
|
|
hrPropagate = ActivateMovr();
|
|
if (!EVAL(SUCCEEDED(hrPropagate)))
|
|
DeactivateMovr(FALSE); // Clean up mess.
|
|
}
|
|
|
|
return hrPropagate;
|
|
}
|
|
|
|
HRESULT CDeskMovr::FreezeEvents(BOOL fFreeze)
|
|
{
|
|
HRESULT hr = IOleControlImpl<CDeskMovr>::FreezeEvents(fFreeze);
|
|
TraceMsg(TF_CUSTOM1, "CDeskMovr::FreezeEvents(fFreeze=%lx) m_nFreezeEvents=%lx; m_fEnabled=%lx, m_bstrTargetName=%ls", (DWORD)fFreeze, m_nFreezeEvents, m_fEnabled, GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
m_nFreezeEvents = fFreeze;
|
|
|
|
if (fFreeze)
|
|
DeactivateMovr(FALSE);
|
|
else
|
|
hr = SmartActivateMovr(hr);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CDeskMovr::Load(LPPROPERTYBAG pPropBag, LPERRORLOG pErrorLog)
|
|
{
|
|
HRESULT hr;
|
|
VARIANT var;
|
|
|
|
ATLTRACE(_T("IPersistPropertyBagImpl::Load\n"));
|
|
|
|
var.vt = VT_BOOL;
|
|
hr = pPropBag->Read(L"Enabled", &var, NULL);
|
|
if (SUCCEEDED(hr) && var.vt == VT_BOOL) {
|
|
m_fEnabled = var.boolVal;
|
|
}
|
|
|
|
var.vt = VT_I4;
|
|
hr = pPropBag->Read(L"Interval", &var, NULL);
|
|
if (SUCCEEDED(hr) && var.vt == VT_I4) {
|
|
m_lInterval = var.lVal;
|
|
}
|
|
|
|
var.vt = VT_BSTR;
|
|
var.bstrVal = NULL;
|
|
hr = pPropBag->Read(L"TargetName", &var, NULL);
|
|
if (SUCCEEDED(hr) && var.vt == VT_BSTR) {
|
|
m_bstrTargetName = var.bstrVal;
|
|
}
|
|
|
|
// This PARAM determines whether the control will be in the
|
|
// windowed or windowless "layer" of the Trident layout.
|
|
var.vt = VT_BOOL;
|
|
hr = pPropBag->Read(L"WindowOnly", &var, NULL);
|
|
if (SUCCEEDED(hr) && var.vt == VT_BOOL) {
|
|
m_bWindowOnly = var.boolVal;
|
|
}
|
|
|
|
hr = _GetZOrderSlot(&m_zIndexTop, TRUE);
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = _GetZOrderSlot(&m_zIndexBottom, FALSE);
|
|
ASSERT(SUCCEEDED(hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CDeskMovr::Save(LPPROPERTYBAG pPropBag, BOOL fClearDirty, BOOL fSaveAllProperties)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
BOOL CDeskMovr::GetCaptionButtonRect(DragMode dm, LPRECT lprc)
|
|
{
|
|
BOOL fSuccess;
|
|
|
|
*lprc = m_rectCaption;
|
|
|
|
switch (dm) {
|
|
case dmClose:
|
|
lprc->left = m_rectCaption.right - (m_cyCaption + m_cxSMBorder);
|
|
fSuccess = (lprc->left > (m_rectCaption.left + m_cyCaption));
|
|
break;
|
|
|
|
case dmMenu:
|
|
lprc->right = lprc->left + (m_cyCaption + m_cxSMBorder);
|
|
fSuccess = (m_rectCaption.right > (m_rectCaption.left + m_cyCaption));
|
|
break;
|
|
|
|
case dmRestore:
|
|
if (ISNORMAL)
|
|
return FALSE;
|
|
else if (ISSPLIT)
|
|
goto CalcSplit;
|
|
else if (ISFULLSCREEN)
|
|
goto CalcFullScreen;
|
|
|
|
ASSERT(FALSE);
|
|
|
|
case dmSplit:
|
|
if (ISSPLIT || !m_fCanResizeX || !m_fCanResizeY)
|
|
{
|
|
return FALSE;
|
|
}
|
|
CalcSplit:
|
|
lprc->left = m_rectCaption.right - (m_cyCaption + m_cxSMBorder);
|
|
OffsetRect(lprc, -(lprc->right - lprc->left), 0);
|
|
fSuccess = (lprc->left > (m_rectCaption.left + 2 * m_cyCaption));
|
|
break;
|
|
|
|
case dmFullScreen:
|
|
if (ISFULLSCREEN || !m_fCanResizeX || !m_fCanResizeY)
|
|
{
|
|
return FALSE;
|
|
}
|
|
CalcFullScreen:
|
|
lprc->left = m_rectCaption.right - (m_cyCaption + m_cxSMBorder);
|
|
OffsetRect(lprc, -((lprc->right - lprc->left) * 2 - 2 * m_cxSMBorder), 0);
|
|
fSuccess = (lprc->left > (m_rectCaption.left + 2 * m_cyCaption));
|
|
break;
|
|
|
|
default:
|
|
ASSERT(FALSE);
|
|
fSuccess = FALSE;
|
|
break;
|
|
}
|
|
|
|
// Shrink the button within the caption and position it adjacent to the border
|
|
if (fSuccess) {
|
|
OffsetRect(lprc, ((dm == dmClose) ? m_cxSMBorder : -m_cxSMBorder), -m_cySMBorder);
|
|
InflateRect(lprc, -m_cxSMBorder, -m_cySMBorder);
|
|
lprc->bottom -= m_cySMBorder; // Take an extra border off the bottom
|
|
}
|
|
|
|
return fSuccess;
|
|
}
|
|
|
|
void CDeskMovr::DrawCaptionButton(HDC hdc, LPRECT lprc, UINT uType, UINT uState, BOOL fErase)
|
|
{
|
|
RECT rcT;
|
|
HRGN hrgnWnd, hrgnRect;
|
|
int iRet;
|
|
|
|
if (fErase)
|
|
FillRect(hdc, lprc, (HBRUSH)(COLOR_3DFACE + 1));
|
|
|
|
rcT = *lprc;
|
|
InflateRect(&rcT, -2 * m_cxSMBorder, -2 * m_cySMBorder);
|
|
|
|
switch (uType) {
|
|
case DMDC_CLOSE:
|
|
uType = DFC_CAPTION;
|
|
goto Draw;
|
|
case DMDC_MENU:
|
|
uType = DFC_SCROLL;
|
|
Draw:
|
|
// We need to clip the border of the outer edge in order to get the drawing effect we
|
|
// want here...
|
|
if (hrgnWnd = CreateRectRgn(0, 0, 0, 0)) {
|
|
if ((iRet = GetClipRgn(hdc, hrgnWnd)) != -1) {
|
|
if (hrgnRect = CreateRectRgnIndirect(&rcT)) {
|
|
SelectClipRgn(hdc, hrgnRect);
|
|
DeleteObject(hrgnRect);
|
|
}
|
|
}
|
|
}
|
|
|
|
DrawFrameControl(hdc, lprc, uType, uState);
|
|
|
|
SelectClipRgn(hdc, (iRet == 1) ? hrgnWnd : NULL);
|
|
if (hrgnWnd)
|
|
DeleteObject(hrgnWnd);
|
|
break;
|
|
|
|
case DMDC_FULLSCREEN:
|
|
case DMDC_SPLIT:
|
|
case DMDC_RESTORE:
|
|
{
|
|
if (uState & DFCS_PUSHED)
|
|
OffsetRect(&rcT, 1, 1);
|
|
|
|
DrawEdge(hdc, &rcT, BDR_OUTER, BF_FLAT | BF_MONO | BF_RECT);
|
|
|
|
#ifndef OLD_CODE
|
|
switch (uType) {
|
|
case DMDC_RESTORE:
|
|
rcT.right = rcT.left + (rcT.right - rcT.left) * 3 / 4;
|
|
rcT.bottom = rcT.top + (rcT.bottom - rcT.top) * 3 / 4;
|
|
rcT.left += (rcT.right - rcT.left) / 2 + 1;
|
|
rcT.top += (rcT.bottom - rcT.top) / 2 + 1;
|
|
FillRect(hdc, &rcT, (HBRUSH)(COLOR_WINDOWFRAME + 1));
|
|
break;
|
|
|
|
case DMDC_SPLIT:
|
|
rcT.top += m_cySMBorder;
|
|
rcT.left += (rcT.right - rcT.left) * 3 / 10;
|
|
DrawEdge(hdc, &rcT, BDR_OUTER, BF_FLAT | BF_MONO | BF_TOP | BF_LEFT);
|
|
break;
|
|
|
|
case DMDC_FULLSCREEN:
|
|
rcT.top += m_cySMBorder;
|
|
DrawEdge(hdc, &rcT, BDR_OUTER, BF_FLAT | BF_MONO | BF_TOP);
|
|
break;
|
|
}
|
|
#else
|
|
switch (uType) {
|
|
case DMDC_RESTORE:
|
|
rcT.right = rcT.left + (rcT.right - rcT.left) * 3 / 4;
|
|
rcT.bottom = rcT.top + (rcT.bottom - rcT.top) * 3 / 4;
|
|
rcT.left += (rcT.right - rcT.left) / 2 + 1;
|
|
rcT.top += (rcT.bottom - rcT.top) / 2 + 1;
|
|
break;
|
|
case DMDC_SPLIT:
|
|
rcT.left += (rcT.right - rcT.left) * 3 / 10;
|
|
break;
|
|
case DMDC_FULLSCREEN:
|
|
break;
|
|
}
|
|
|
|
FillRect(hdc, &rcT, (HBRUSH)(COLOR_WINDOWFRAME + 1));
|
|
#endif
|
|
}
|
|
break;
|
|
}
|
|
|
|
// DFCS_FLAT means no border to us
|
|
if (!(uState & DFCS_FLAT))
|
|
DrawEdge(hdc, lprc, ((uState & DFCS_PUSHED) ? BDR_SUNKENOUTER : BDR_RAISEDINNER), BF_RECT);
|
|
}
|
|
|
|
void CDeskMovr::DrawCaption(HDC hdc, UINT uDrawFlags, int x, int y)
|
|
{
|
|
RECT rect;
|
|
UINT uState;
|
|
DragMode dmT;
|
|
|
|
// Draw the caption
|
|
if (uDrawFlags & DMDC_CAPTION) {
|
|
rect = m_rectCaption;
|
|
OffsetRect(&rect, x, y);
|
|
FillRect(hdc, &rect, (HBRUSH)(COLOR_3DFACE + 1));
|
|
}
|
|
|
|
// Draw the caption frame controls
|
|
for (dmT = dmMenu; dmT < dmMove; dmT = (DragMode)((int)dmT + 1))
|
|
{
|
|
if ((uDrawFlags & DMDCFROMDM(dmT)) && GetCaptionButtonRect(dmT, &rect))
|
|
{
|
|
if (dmT == dmMenu)
|
|
uState = DFCS_SCROLLDOWN;
|
|
else if (dmT == dmClose)
|
|
uState = DFCS_CAPTIONCLOSE;
|
|
else
|
|
uState = 0;
|
|
|
|
if ((dmT == dmClose) && SHRestricted(REST_NOCLOSEDESKCOMP))
|
|
uState |= DFCS_INACTIVE | DFCS_FLAT;
|
|
else
|
|
{
|
|
if (m_CaptionState & PUSHED(dmT))
|
|
uState |= DFCS_PUSHED;
|
|
if (!(m_CaptionState & (TRACKED(dmT) | PUSHED(dmT))))
|
|
uState |= DFCS_FLAT;
|
|
}
|
|
OffsetRect(&rect, x, y);
|
|
DrawCaptionButton(hdc, &rect, DMDCFROMDM(dmT), uState, !(uDrawFlags & DMDC_CAPTION));
|
|
}
|
|
}
|
|
}
|
|
|
|
HRESULT CDeskMovr::OnDraw(ATL_DRAWINFO& di)
|
|
{
|
|
RECT& rc = *(RECT*)di.prcBounds;
|
|
|
|
RECT r;
|
|
HBRUSH hbrush = (HBRUSH)(COLOR_3DFACE + 1);
|
|
|
|
// top edge
|
|
r.left = rc.left;
|
|
r.top = rc.top;
|
|
r.right = rc.right;
|
|
r.bottom = rc.top + m_cyBorder;
|
|
FillRect(di.hdcDraw, &r, hbrush);
|
|
// left edge
|
|
r.top = rc.top + m_cyBorder;
|
|
r.right = rc.left + m_cxBorder;
|
|
r.bottom = rc.bottom - m_cyBorder;
|
|
FillRect(di.hdcDraw, &r, hbrush);
|
|
// right edge
|
|
r.right = rc.right;
|
|
r.left = rc.right - m_cxBorder;
|
|
FillRect(di.hdcDraw, &r, hbrush);
|
|
// bottom edge
|
|
r.left = rc.left;
|
|
r.top = rc.bottom - m_cyBorder;
|
|
r.right = rc.right;
|
|
r.bottom = rc.bottom;
|
|
FillRect(di.hdcDraw, &r, hbrush);
|
|
|
|
if (m_cyCaption != 0) {
|
|
DrawCaption(di.hdcDraw, DMDC_ALL, rc.left, rc.top);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CDeskMovr::GetParentWindow(void)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (!m_hwndParent)
|
|
{
|
|
if (m_spInPlaceSite)
|
|
hr = m_spInPlaceSite->GetWindow(&m_hwndParent);
|
|
else
|
|
{
|
|
IOleInPlaceSiteWindowless * poipsw;
|
|
|
|
ASSERT(m_spClientSite);
|
|
if (m_spClientSite &&
|
|
SUCCEEDED(hr = m_spClientSite->QueryInterface(IID_IOleInPlaceSiteWindowless, (void **)&poipsw)))
|
|
{
|
|
hr = poipsw->GetWindow(&m_hwndParent);
|
|
poipsw->Release();
|
|
}
|
|
}
|
|
|
|
if (!m_hwndParent)
|
|
hr = S_FALSE; // We failed to get it.
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
void CDeskMovr::DeactivateMovr(BOOL fDestroy)
|
|
{
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::DeactivateMovr() m_fTimer=%lx, m_bstrTargetName=%ls", m_fTimer, GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
if (fDestroy || (m_uiTimerID == DETECT_TIMER_ID)) {
|
|
if (m_fTimer)
|
|
{
|
|
m_TimerWnd.KillTimer(m_uiTimerID);
|
|
m_fTimer = FALSE;
|
|
}
|
|
if (m_TimerWnd.m_hWnd)
|
|
m_TimerWnd.DestroyWindow();
|
|
#ifndef SHDOC401_DLL
|
|
if (m_uiTimerID == ANIMATE_TIMER_ID)
|
|
g_fAnimTimer = FALSE;
|
|
#endif
|
|
}
|
|
|
|
// DismissSelfNow();
|
|
|
|
ATOMICRELEASE(m_pistyle);
|
|
ATOMICRELEASE(m_pistyleTarget);
|
|
ATOMICRELEASE(m_pielemTarget);
|
|
|
|
_ChangeCapture(FALSE);
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::ActivateMovr()
|
|
{
|
|
HRESULT hr;
|
|
|
|
// flush out old interface pointers
|
|
DeactivateMovr(FALSE);
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::ActivateMovr() m_fTimer=%lx, m_bstrTargetName=%ls", m_fTimer, GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
if (m_fEnabled && SUCCEEDED(hr = GetOurStyle()))
|
|
{
|
|
if ((m_bstrTargetName != NULL) && (m_lInterval > 0))
|
|
{
|
|
if (!m_TimerWnd.m_hWnd)
|
|
{
|
|
// create a new timer.
|
|
RECT rc = { 0, 0, 0 , 0 };
|
|
|
|
// We attempt to get our parent HWND (m_hwndParent) now.
|
|
// If we fail (and we will sometimes), then we will get it later when
|
|
// we need it and Trident is then ready.
|
|
GetParentWindow();
|
|
|
|
m_TimerWnd.Create(NULL, rc, _T("Timer"), WS_POPUP);
|
|
}
|
|
if (!m_fTimer)
|
|
m_fTimer = m_TimerWnd.SetTimer(m_uiTimerID, m_lInterval, NULL) != 0;
|
|
}
|
|
else
|
|
{
|
|
#ifdef HIDE_ALL_HANDLES
|
|
hr = S_FALSE;
|
|
#else
|
|
hr = E_FAIL;
|
|
#endif
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CDeskMovr::GetOurStyle(void)
|
|
{
|
|
HRESULT hr;
|
|
IOleControlSite *pictlsite = 0;
|
|
IDispatch *pidisp = 0;
|
|
|
|
// Reach up to get our extender, who is the custodian of our element style
|
|
if (m_spClientSite &&
|
|
EVAL(SUCCEEDED(hr = m_spClientSite->QueryInterface(IID_IOleControlSite, (LPVOID*)&pictlsite))) &&
|
|
EVAL(SUCCEEDED(hr = pictlsite->GetExtendedControl(&pidisp))))
|
|
{
|
|
DISPPARAMS dispparamsNoArgs = { NULL, NULL, 0, 0 };
|
|
VARIANT var;
|
|
|
|
VariantInit(&var);
|
|
|
|
// Alas, all we have is IDispatch on our extender, so we'll have to use Invoke to get
|
|
// the style object...
|
|
hr = pidisp->Invoke(DISPID_IHTMLELEMENT_STYLE, IID_NULL,
|
|
LOCALE_USER_DEFAULT, DISPATCH_PROPERTYGET,
|
|
&dispparamsNoArgs, &var, NULL, NULL);
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
if (var.vt == VT_DISPATCH)
|
|
hr = var.pdispVal->QueryInterface(IID_IHTMLStyle, (LPVOID*)&m_pistyle);
|
|
else
|
|
hr = E_FAIL; // Try VariantChangeType?????
|
|
|
|
VariantClear(&var);
|
|
}
|
|
}
|
|
|
|
ATOMICRELEASE(pictlsite);
|
|
ATOMICRELEASE(pidisp);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
void CDeskMovr::UpdateCaption(UINT uDrawFlags)
|
|
{
|
|
HDC hdc;
|
|
int x = 0, y = 0;
|
|
|
|
if (m_bWndLess) {
|
|
if (!m_spInPlaceSite || !SUCCEEDED(m_spInPlaceSite->GetDC(NULL, 0, &hdc)))
|
|
return;
|
|
}
|
|
else {
|
|
hdc = ::GetDC(m_hWnd);
|
|
}
|
|
|
|
_MapPoints(&x, &y);
|
|
|
|
DrawCaption(hdc, uDrawFlags, -x, -y);
|
|
|
|
if (m_bWndLess) {
|
|
m_spInPlaceSite->ReleaseDC(hdc);
|
|
}
|
|
else {
|
|
::ReleaseDC(m_hWnd, hdc);
|
|
}
|
|
}
|
|
|
|
void CDeskMovr::CheckCaptionState(int x, int y)
|
|
{
|
|
DragMode dm, dmT;
|
|
UINT uDrawFlags = 0;
|
|
|
|
_MapPoints(&x, &y);
|
|
|
|
POINT pt = { x, y };
|
|
|
|
if (m_fCaptured)
|
|
dm = dmNull;
|
|
else
|
|
dm = DragModeFromPoint(pt);
|
|
|
|
if (dm >= dmMenu && dm < dmMove)
|
|
{
|
|
if (!(m_CaptionState & (PUSHED(dm) | TRACKED(dm))))
|
|
{
|
|
m_CaptionState |= TRACKED(dm);
|
|
uDrawFlags |= DMDCFROMDM(dm);
|
|
}
|
|
}
|
|
|
|
for (dmT = dmMenu; dmT < dmMove; dmT = (DragMode)((int)dmT + 1))
|
|
{
|
|
if (dm != dmT && (m_CaptionState & (PUSHED(dmT) | TRACKED(dmT))))
|
|
{
|
|
m_CaptionState &= ~(PUSHED(dmT) | TRACKED(dmT));
|
|
uDrawFlags |= DMDCFROMDM(dmT);
|
|
}
|
|
}
|
|
|
|
if (uDrawFlags)
|
|
UpdateCaption(uDrawFlags);
|
|
}
|
|
|
|
|
|
// CDeskMovr::DoMouseDown [instance method]
|
|
|
|
// Respond to mouse down messages in our control. Initiate move/resize.
|
|
|
|
// Parameters:
|
|
// int - [in] mouse message key flags
|
|
// int - [in] mouse x location in control coords
|
|
// int - [in] mouse y location in control coords
|
|
|
|
// Output:
|
|
// <none>
|
|
|
|
// Notes:
|
|
BOOL CDeskMovr::HandleNonMoveSize(DragMode dm)
|
|
{
|
|
m_dmCur = dm;
|
|
switch (dm) {
|
|
case dmMenu:
|
|
case dmClose:
|
|
case dmRestore:
|
|
case dmFullScreen:
|
|
case dmSplit:
|
|
if (m_dmCur != dmClose || !SHRestricted(REST_NOCLOSEDESKCOMP)) // Special case for Close, check restriction
|
|
{
|
|
m_CaptionState &= ~(TRACKED(m_dmCur));
|
|
m_CaptionState |= PUSHED(m_dmCur);
|
|
UpdateCaption(DMDCFROMDM(m_dmCur));
|
|
// Perform the operation on the up-click of the mouse...
|
|
}
|
|
|
|
if (m_dmCur == dmMenu && EVAL(S_OK == GetParentWindow())) // Special case for Menu, invoke on the down click
|
|
{
|
|
_DisplayContextMenu();
|
|
}
|
|
return TRUE;
|
|
break;
|
|
|
|
default:
|
|
return FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
LRESULT CDeskMovr::OnMouseDown(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
int x = (short)LOWORD(lParam);
|
|
int y = (short)HIWORD(lParam);
|
|
|
|
_MapPoints(&x, &y);
|
|
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::OnMouseDown() Mouse=<%d,%d>, Inner=<%d,%d,%d,%d>, Caption=<%d,%d,%d,%d>, m_bstrTargetName=%ls",
|
|
x, y, m_rectInner.left, m_rectInner.top, m_rectInner.right, m_rectInner.bottom,
|
|
m_rectCaption.left, m_rectCaption.top, m_rectCaption.right, m_rectCaption.bottom, GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
POINT pt = { x, y };
|
|
m_dmCur = DragModeFromPoint(pt);
|
|
|
|
if (HandleNonMoveSize(m_dmCur))
|
|
return 0;
|
|
|
|
switch (m_dmCur) {
|
|
case dmMove:
|
|
m_dx = -x;
|
|
m_dy = -y;
|
|
break;
|
|
|
|
case dmSizeWHBR:
|
|
m_dx = m_rectInner.right - x;
|
|
m_dy = m_rectInner.bottom - y;
|
|
break;
|
|
case dmSizeWHTL:
|
|
m_dx = m_rectInner.left - x;
|
|
m_dy = m_rectInner.top + m_cyCaption - y;
|
|
break;
|
|
case dmSizeWHTR:
|
|
m_dx = m_rectInner.right - x;
|
|
m_dy = m_rectInner.top + m_cyCaption - y;
|
|
break;
|
|
case dmSizeWHBL:
|
|
m_dx = m_rectInner.left - x;
|
|
m_dy = m_rectInner.bottom - y;
|
|
break;
|
|
case dmSizeTop:
|
|
m_dx = 0;
|
|
m_dy = m_rectInner.top + m_cyCaption - y;
|
|
break;
|
|
case dmSizeBottom:
|
|
m_dx = 0;
|
|
m_dy = m_rectInner.bottom - y;
|
|
break;
|
|
case dmSizeLeft:
|
|
m_dx = m_rectInner.left - x;
|
|
m_dy = 0;
|
|
break;
|
|
case dmSizeRight:
|
|
m_dx = m_rectInner.right - x;
|
|
m_dy = 0;
|
|
break;
|
|
default:
|
|
bHandled = FALSE;
|
|
return 1;
|
|
}
|
|
|
|
#ifdef DEBUG
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::OnMouseDown() New DragMode=""%s""", g_szDragModeStr[m_dmCur]);
|
|
#endif // DEBUG
|
|
|
|
//BUGBUG: (seanf, 1/31/97) temporary defense against 17902. We really
|
|
// shouldn't ever be in visible and non-targeted at the same time, but
|
|
// the resize trick we pull in CDeskMovr::ActivateMovr() to get us
|
|
// in-place active exposes a 1X1 pixel area, just big enough for StanTak
|
|
// to click on when we don't have a target, which then kills us when
|
|
// we try to move the non-existent target.
|
|
if (m_pielemTarget != NULL) {
|
|
_ChangeCapture(TRUE);
|
|
|
|
if (m_fCaptured)
|
|
{
|
|
// Move the target to the top and put ourselves just under it
|
|
VARIANT varZ;
|
|
|
|
m_pistyleTarget->get_zIndex(&varZ);
|
|
|
|
// Does the component need to be moved to the top?
|
|
if (!CAPTION_ONLY && ((VT_I4 != varZ.vt) || (varZ.lVal != m_zIndexTop)))
|
|
{
|
|
// Yes.
|
|
varZ.vt = VT_I4;
|
|
varZ.lVal = ++m_zIndexTop;
|
|
// Move the DeskMover ActiveX Control on top of everything.
|
|
m_pistyle->put_zIndex(varZ);
|
|
|
|
// Move the Desktop Item on top of the DeskMover
|
|
varZ.lVal = ++m_zIndexTop;
|
|
m_pistyleTarget->put_zIndex(varZ);
|
|
}
|
|
}
|
|
#ifdef DEBUG
|
|
if (!m_fCaptured)
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::OnMouseDown() Unable to get capture, tracking will fail!");
|
|
#endif
|
|
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
// CDeskMovr::DoMouseUp [instance method]
|
|
|
|
// Respond to mouse down messages in our control. Terminate move/resize.
|
|
|
|
// Parameters:
|
|
// int - [in] mouse message key flags
|
|
// int - [in] mouse x location in control coords
|
|
// int - [in] mouse y location in control coords
|
|
// UINT - [in] from the DeskMovrParts enum
|
|
|
|
// Output:
|
|
// <none>
|
|
|
|
// Notes:
|
|
|
|
LRESULT CDeskMovr::OnMouseUp(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
if (m_fCaptured) {
|
|
PersistTargetPosition(m_pielemTarget, m_left, m_top, m_width, m_height, m_zIndexTop, FALSE, FALSE, m_ItemState);
|
|
_ChangeCapture(FALSE);
|
|
}
|
|
else {
|
|
int x = (short)LOWORD(lParam);
|
|
int y = (short)HIWORD(lParam);
|
|
|
|
_MapPoints(&x, &y);
|
|
|
|
POINT pt = { x, y };
|
|
DragMode dm = DragModeFromPoint(pt);
|
|
|
|
if ((dm >= dmMenu) && (dm < dmMove) && (m_CaptionState & PUSHED(dm)))
|
|
{
|
|
m_CaptionState &= ~(PUSHED(dm));
|
|
m_CaptionState |= TRACKED(dm);
|
|
UpdateCaption(DMDCFROMDM(dm));
|
|
|
|
switch (dm) {
|
|
case dmClose:
|
|
// AnimateToTray(m_hwndParent, m_left, m_top, m_width, m_height);
|
|
IElemCloseDesktopComp(m_pielemTarget);
|
|
break;
|
|
|
|
case dmRestore:
|
|
_HandleZoom(IDM_DCCM_RESTORE);
|
|
break;
|
|
|
|
case dmFullScreen:
|
|
_HandleZoom(IDM_DCCM_FULLSCREEN);
|
|
break;
|
|
|
|
case dmSplit:
|
|
_HandleZoom(IDM_DCCM_SPLIT);
|
|
break;
|
|
}
|
|
|
|
if (dm != dmMenu)
|
|
DismissSelfNow();
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
// CDeskMovrControl::DoMouseMove [instance method]
|
|
|
|
// Respond to mouse move messages in our control and when moving/sizing.
|
|
|
|
// Parameters:
|
|
|
|
// Output:
|
|
// <none>
|
|
|
|
// Notes:
|
|
|
|
LRESULT CDeskMovr::OnPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::OnPaint() uMsg=%lx, wParam=%lx, lParam=%lx, m_bstrTargetName=%ls", uMsg, wParam, lParam, GEN_DEBUGSTRW(m_bstrTargetName));
|
|
return CComControl<CDeskMovr>::OnPaint(uMsg, wParam, lParam, bHandled);
|
|
}
|
|
|
|
|
|
// CDeskMovrControl::DoMouseMove [instance method]
|
|
|
|
// Respond to mouse move messages in our control and when moving/sizing.
|
|
|
|
// Parameters:
|
|
|
|
// Output:
|
|
// <none>
|
|
|
|
// Notes:
|
|
|
|
LRESULT CDeskMovr::OnMouseMove(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
CheckCaptionState((short)LOWORD(lParam), (short)HIWORD(lParam));
|
|
|
|
if (m_fCaptured && EVAL(S_OK == GetParentWindow()))
|
|
{
|
|
// Okay, it's a hit on one of our gadgets.
|
|
// We're only interested in mouse moves and mouse ups if we're in the
|
|
// process of a drag or resize
|
|
HRESULT hr;
|
|
POINT ptDoc; // location in document window coords
|
|
POINT ptScreen;
|
|
HWND hwndParent = m_hwndParent;
|
|
|
|
int x = (short)LOWORD(lParam);
|
|
int y = (short)HIWORD(lParam);
|
|
|
|
ptScreen.x = x;
|
|
ptScreen.y = y;
|
|
ptDoc = ptScreen;
|
|
if (!m_bWndLess)
|
|
::MapWindowPoints(m_hWnd, hwndParent, &ptDoc, 1);
|
|
|
|
if (m_dmCur == dmMove)
|
|
hr = MoveSelfAndTarget(ptDoc.x + m_dx + m_cxBorder, ptDoc.y + m_dy + m_cyBorder + m_cyCaption);
|
|
else if (m_dmCur > dmMove)
|
|
hr = SizeSelfAndTarget(ptDoc);
|
|
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
|
|
// Set m_cSkipTimer so that we delay dismissing the mover...
|
|
m_cSkipTimer = GET_SKIP_COUNT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
LRESULT CDeskMovr::OnTimer(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
HRESULT hr;
|
|
IHTMLElement *pielem;
|
|
POINT ptCursor;
|
|
BOOL fDidWork = FALSE;
|
|
|
|
#ifndef SHDOC401_DLL
|
|
/*
|
|
* Check our animation timer first. If we are able to disable animations then
|
|
* blow away the timer. Otherwise reset the timer for 60 seconds and keep on
|
|
* looking.
|
|
*/
|
|
if (wParam == ANIMATE_TIMER_ID)
|
|
{
|
|
if (CombView_EnableAnimations(FALSE))
|
|
{
|
|
m_TimerWnd.SetTimer(ANIMATE_TIMER_ID, ANIMATE_TIMER_INTERVAL, NULL);
|
|
}
|
|
else
|
|
{
|
|
m_TimerWnd.KillTimer(m_uiTimerID);
|
|
m_uiTimerID = DETECT_TIMER_ID;
|
|
g_fAnimTimer = FALSE;
|
|
m_fTimer = FALSE;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
if (!m_fTimer || g_fIgnoreTimers || !GetCursorPos(&ptCursor) || !m_pistyle)
|
|
return 0;
|
|
|
|
if (ptCursor.x == m_ptMouseCursor.x && ptCursor.y == m_ptMouseCursor.y)
|
|
// Mouse stayed still from last time we did a timer so, do nothing
|
|
return 0;
|
|
|
|
pielem = NULL;
|
|
|
|
if (EVAL(S_OK == GetParentWindow()))
|
|
{
|
|
HWND hwndParent = m_hwndParent;
|
|
HWND hwndCursor = WindowFromPoint(ptCursor);
|
|
|
|
if ((hwndCursor != hwndParent) && !::IsChild(hwndParent, hwndCursor))
|
|
{
|
|
// The mouse has drifted out of our window, so lose our target, if any
|
|
if (m_iSrcTarget >= 0)
|
|
{
|
|
hr = MoveSelfToTarget(NULL, NULL);
|
|
ASSERT(SUCCEEDED(hr));
|
|
if (hr != S_FALSE)
|
|
{
|
|
fDidWork = TRUE;
|
|
}
|
|
}
|
|
if (GetCurrentThreadId() == g_dwHookThreadId) {
|
|
#ifndef SHDOC401_DLL
|
|
// Set ourselves up so we can look to see if our animations can be turned off
|
|
if (!g_fAnimTimer)
|
|
{
|
|
if (m_fTimer)
|
|
m_TimerWnd.KillTimer(m_uiTimerID);
|
|
|
|
if (g_fAnimTimer = (m_TimerWnd.SetTimer(ANIMATE_TIMER_ID, ANIMATE_TIMER_INTERVAL / 10, NULL) != 0))
|
|
m_uiTimerID = ANIMATE_TIMER_ID;
|
|
m_fTimer = g_fAnimTimer;
|
|
}
|
|
#endif
|
|
DismissSelfNow();
|
|
DeactivateMovr(FALSE);
|
|
if (!g_hMouseHook)
|
|
g_hMouseHook = SetWindowsHookEx(WH_MOUSE, DeskMovr_MouseHook, NULL, GetCurrentThreadId());
|
|
}
|
|
}
|
|
else if (!(GetDesktopFlags() & COMPONENTS_LOCKED) && SUCCEEDED(hr = _IsInElement(hwndParent, &ptCursor, &pielem)))
|
|
{
|
|
// See if we need to do anything based on the element under the mouse pointer
|
|
hr = _TrackElement(&ptCursor, pielem, &fDidWork);
|
|
// we're done with this particular interface pointer
|
|
pielem->Release();
|
|
}
|
|
else if (m_iSrcTarget != -1) {
|
|
// Check to see if we should expand border to size border width
|
|
if (TrackCaption(&ptCursor))
|
|
{
|
|
TrackTarget(NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!fDidWork)
|
|
m_ptMouseCursor = ptCursor;
|
|
|
|
return 0;
|
|
}
|
|
|
|
LRESULT CDeskMovr::OnEraseBkgnd(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
if (!m_bWndLess) {
|
|
RECT rc;
|
|
::GetClientRect(m_hWnd, &rc);
|
|
FillRect((HDC)wParam, &rc, (HBRUSH)(COLOR_3DFACE + 1));
|
|
}
|
|
bHandled = TRUE;
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
// DismissSelfNow - Little helper function to dismiss the mover immediately
|
|
|
|
// Normally dismissal of the mover is desired to be done on a delayed basis. However,
|
|
// there are situations such as when the user clicks on UI or capture is lost etc. where
|
|
// it is desirable to dismiss the mover immediately.
|
|
|
|
void CDeskMovr::DismissSelfNow(void)
|
|
{
|
|
HRESULT hr;
|
|
m_cSkipTimer = 0;
|
|
hr = MoveSelfToTarget(NULL, NULL);
|
|
ASSERT(SUCCEEDED(hr) && (hr != S_FALSE));
|
|
}
|
|
|
|
LRESULT CDeskMovr::OnCaptureChanged(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
if (m_fCaptured) {
|
|
_ChangeCapture(FALSE);
|
|
PersistTargetPosition(m_pielemTarget, m_left, m_top, m_width, m_height, m_zIndexTop, FALSE, FALSE, m_ItemState);
|
|
DismissSelfNow();
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
HRESULT CDeskMovr::InPlaceDeactivate(void)
|
|
{
|
|
DeactivateMovr(FALSE);
|
|
TraceMsg(TF_CUSTOM1, "CDeskMovr::InPlaceDeactivate()");
|
|
return CComControl<CDeskMovr>::IOleInPlaceObject_InPlaceDeactivate();
|
|
}
|
|
|
|
LRESULT CDeskMovr::OnSetCursor(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
|
|
{
|
|
if (EVAL(S_OK == GetParentWindow()))
|
|
{
|
|
POINT ptCursor;
|
|
DragMode dm;
|
|
|
|
GetCursorPos(&ptCursor);
|
|
|
|
::ScreenToClient(m_hwndParent, &ptCursor);
|
|
|
|
// Get ptCursor into deskmovr local coords
|
|
ptCursor.x -= m_left - (CAPTION_ONLY ? 0 : m_cxBorder);
|
|
ptCursor.y -= m_top - (CAPTION_ONLY ? 0 : (m_cyBorder + m_cyCaption));
|
|
|
|
dm = DragModeFromPoint(ptCursor);
|
|
m_hcursor = CursorFromDragMode(dm);
|
|
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::OnSetCursor() Mouse=<%d,%d>, Inner=<%d,%d,%d,%d>, Caption=<%d,%d,%d,%d>, m_bstrTargetName=%ls",
|
|
ptCursor.x, ptCursor.y, m_rectInner.left, m_rectInner.top, m_rectInner.right, m_rectInner.bottom,
|
|
m_rectCaption.left, m_rectCaption.top, m_rectCaption.right, m_rectCaption.bottom, GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
#ifdef DEBUG
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::OnSetCursor() New DragMode=""%s""", g_szDragModeStr[dm]);
|
|
#endif // DEBUG
|
|
|
|
if (EVAL(m_hcursor != NULL))
|
|
SetCursor(m_hcursor);
|
|
else
|
|
bHandled = FALSE;
|
|
}
|
|
|
|
return !bHandled;
|
|
}
|
|
|
|
|
|
void CDeskMovr::TrackTarget(POINT * pptDoc)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
if (m_fEnabled && m_pielemTarget != NULL) {
|
|
LONG left, top;
|
|
POINT pt;
|
|
VARIANT varZ;
|
|
COMPSTATEINFO CompState;
|
|
|
|
varZ.vt = VT_I4;
|
|
|
|
CLEANUP_ON_FAILURE(hr = CSSOM_TopLeft(m_pielemTarget, &pt));
|
|
m_top = pt.y;
|
|
m_left = pt.x;
|
|
|
|
CLEANUP_ON_FAILURE(hr = m_pielemTarget->get_offsetHeight(&m_height));
|
|
CLEANUP_ON_FAILURE(hr = m_pielemTarget->get_offsetWidth(&m_width));
|
|
|
|
// Hack so we don't get weird painting effect of the window hopping to the new
|
|
// target with the old target's size.
|
|
if (!m_bWndLess && m_cyCaption == 0)
|
|
::SetWindowPos(m_hWnd, NULL, 0, 0, 0, 0, SWP_NOZORDER | SWP_NOACTIVATE);
|
|
|
|
// Get our rectangle synced with the target (so TrackCaption works properly)
|
|
SyncRectsToTarget();
|
|
// If we discover we want to display the size-border or caption
|
|
// right now then we need to recalculate our rects.
|
|
if (pptDoc && TrackCaption(pptDoc))
|
|
SyncRectsToTarget();
|
|
|
|
CLEANUP_ON_FAILURE(hr = m_pistyleTarget->get_zIndex(&varZ));
|
|
if (!CAPTION_ONLY || (m_cxBorder == m_cxSMBorder))
|
|
--varZ.lVal;
|
|
else
|
|
++varZ.lVal;
|
|
CLEANUP_ON_FAILURE(hr = SafeZOrderSet(m_pistyle, varZ.lVal));
|
|
|
|
// BUGBUG: If this is hosted in a window that has scrollbars,
|
|
// we don't correctly add the screen to document offset
|
|
// when changing the location of the component.
|
|
// This causes us to drag incorrectly.
|
|
|
|
// 98/10/02 #176729 vtan: Now uses the component left and top to
|
|
// position the caption. Offset the caption if the component is
|
|
// not zoomed. If zoomed then just draw over the component.
|
|
|
|
left = m_left;
|
|
top = m_top;
|
|
if (!CAPTION_ONLY)
|
|
{
|
|
left -= m_cxBorder;
|
|
top -= m_cyBorder;
|
|
top -= m_cyCaption;
|
|
}
|
|
hr = m_pistyle->put_pixelLeft(left);
|
|
hr = m_pistyle->put_pixelTop(top);
|
|
hr = m_pistyle->put_pixelWidth(m_rectOuter.right);
|
|
hr = m_pistyle->put_pixelHeight(m_rectOuter.bottom);
|
|
|
|
hr = m_pistyle->put_visibility((BSTR)s_sstrVisible.wsz);
|
|
|
|
// We need to persist the original state of the item out now if the item's current width/height is -1
|
|
// This occurs when we are fitting an image to it's default size, we need to make sure the
|
|
// original size real values so it works properly.
|
|
ObtainSavedStateForElem(m_pielemTarget, &CompState, FALSE);
|
|
if (m_bWndLess && CompState.dwWidth == COMPONENT_DEFAULT_WIDTH && CompState.dwHeight == COMPONENT_DEFAULT_HEIGHT)
|
|
PersistTargetPosition(m_pielemTarget, m_left, m_top, m_width, m_height, varZ.lVal, FALSE, TRUE, CompState.dwItemState);
|
|
}
|
|
|
|
CleanUp:
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
|
|
BOOL CDeskMovr::TrackCaption(POINT *pptDoc)
|
|
{
|
|
int cyCaption, cyCaptionNew;
|
|
POINT ptMovr;
|
|
DragMode dmNew;
|
|
BOOL fRetVal = FALSE;
|
|
|
|
//TraceMsg(TF_CUSTOM2, "CDeskMovr::TrackCaption() Mouse=<%d,%d>", ptMovr.x, ptMovr.y);
|
|
|
|
if (pptDoc)
|
|
{
|
|
ptMovr = *pptDoc;
|
|
// need a hit test of some sort within the deskmovr to control border swelling
|
|
ptMovr.x -= m_left - m_cxBorder;
|
|
ptMovr.y -= m_top - (m_cyBorder + m_cyCaption);
|
|
|
|
dmNew = DragModeFromPoint(ptMovr);
|
|
|
|
cyCaption = GET_CYCAPTION;
|
|
|
|
if (dmNew == dmNull) {
|
|
BOOL fInner;
|
|
int iInflate;
|
|
RECT rc;
|
|
// Treat something near the size border as a size border hit
|
|
// so we expand to the size border as the user nears the edge.
|
|
fInner = PtInRect(&m_rectInner, ptMovr);
|
|
|
|
if (fInner) {
|
|
rc = m_rectInner;
|
|
iInflate = -cyCaption;
|
|
}
|
|
else {
|
|
rc = m_rectOuter;
|
|
iInflate = cyCaption;
|
|
}
|
|
|
|
InflateRect(&rc, iInflate, iInflate);
|
|
if (fInner != PtInRect(&rc, ptMovr))
|
|
dmNew = dmSizeRight;
|
|
}
|
|
|
|
if ((pptDoc->y >= m_top - (m_cyBorder + 2 * m_cyCaption) &&
|
|
pptDoc->y <= (m_top + CAPTIONBAR_HOTAREA(cyCaption, m_cyCaption))))
|
|
cyCaptionNew = cyCaption;
|
|
else
|
|
cyCaptionNew = 0;
|
|
}
|
|
else
|
|
{
|
|
cyCaptionNew = GET_CYCAPTION;
|
|
dmNew = dmSizeRight;
|
|
}
|
|
|
|
if (cyCaptionNew != m_cyCaption ||
|
|
(m_dmTrack != dmNew && !((m_dmTrack > dmMove) && (dmNew > dmMove)))) {
|
|
m_cyCaption = cyCaptionNew;
|
|
if (m_cyCaption == 0)
|
|
m_CaptionState = 0;
|
|
m_dmTrack = dmNew;
|
|
fRetVal = TRUE;
|
|
}
|
|
else
|
|
m_cyCaption = cyCaptionNew;
|
|
|
|
return fRetVal;
|
|
}
|
|
|
|
int CDeskMovr::CountActiveCaptions()
|
|
{
|
|
int iCount = 0;
|
|
|
|
if (g_dwHookThreadId == GetCurrentThreadId())
|
|
{
|
|
for (int i = 0; i < CDESKMOVR_TRACK_COUNT; i++) {
|
|
if (g_apDM[i] && g_apDM[i]->m_pistyleTarget)
|
|
iCount++;
|
|
}
|
|
}
|
|
return iCount;
|
|
}
|
|
|
|
HRESULT CDeskMovr::_TrackElement(POINT * ppt, IHTMLElement * pielem, BOOL * fDidWork)
|
|
{
|
|
HRESULT hr;
|
|
IHTMLElement *pTargElem = NULL;
|
|
LONG iSrcTarget = -1;
|
|
|
|
ASSERT(pielem);
|
|
|
|
if (FFindTargetElement(pielem, &pTargElem))
|
|
{
|
|
hr = pTargElem->get_sourceIndex(&iSrcTarget);
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
|
|
// If the m_iSrcTarget isn't the same as the SrcTarget under our cursor,
|
|
// then we should move on top of it.
|
|
if (m_iSrcTarget != iSrcTarget)
|
|
{
|
|
*fDidWork = TRUE;
|
|
|
|
if ((CountActiveCaptions() > 1) && (-1 == iSrcTarget))
|
|
m_cSkipTimer = 0;
|
|
|
|
// Yes, we need to move on top of it.
|
|
hr = MoveSelfToTarget(pTargElem, ppt);
|
|
ASSERT(SUCCEEDED(hr));
|
|
if (hr != S_FALSE)
|
|
m_iSrcTarget = iSrcTarget;
|
|
}
|
|
else
|
|
{
|
|
// No, so that means we already have focus...
|
|
if (ppt && TrackCaption(ppt))
|
|
{
|
|
TrackTarget(NULL);
|
|
}
|
|
}
|
|
|
|
if (pTargElem != NULL) {
|
|
pTargElem->Release(); // MoveSelfToTarget will have secured our reference
|
|
}
|
|
|
|
hr = (m_iSrcTarget == -1) ? S_FALSE : S_OK;
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// CDeskMovr::InitAttributes [instance method]
|
|
|
|
// Finds out if the element is resizeable in X and Y direction and sets the
|
|
// BITBOOLs accordingly.
|
|
|
|
// Also determines what state the element is in and sets m_ItemState.
|
|
|
|
// Parameters:
|
|
// IHTMLElement* [in] - interface on event source element
|
|
|
|
// Output:
|
|
// HRESULT - various. S_OK if operation succeeded.
|
|
|
|
HRESULT CDeskMovr::InitAttributes(IHTMLElement *pielem)
|
|
{
|
|
HRESULT hr;
|
|
TCHAR szMember[MAX_ID_LENGTH];
|
|
|
|
ASSERT(pielem);
|
|
|
|
m_fCanResizeX = m_fCanResizeY = FALSE; //Assume "Can't resize!
|
|
|
|
// The resizeable member is not required to be specified, only override defaults if present.
|
|
if (SUCCEEDED(GetHTMLElementStrMember(pielem, szMember, SIZECHARS(szMember), (BSTR)(s_sstrResizeableMember.wsz))))
|
|
{
|
|
if (StrChr(szMember, TEXT('X')))
|
|
m_fCanResizeX = TRUE;
|
|
|
|
if (StrChr(szMember, TEXT('Y')))
|
|
m_fCanResizeY = TRUE;
|
|
}
|
|
|
|
// The ItemState is required, return failure if we fail to find the ID
|
|
if (SUCCEEDED(hr = GetHTMLElementStrMember(pielem, szMember, SIZECHARS(szMember), (BSTR)(s_sstrIDMember.wsz))))
|
|
m_ItemState = GetCurrentState(szMember);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// CDeskMovr::MoveSelfToTarget [instance method]
|
|
|
|
// Handles Trident document events as mouse moves over the desktop.
|
|
|
|
// Parameters:
|
|
// IHTMLElement* [in] - interface on event source element
|
|
// POINT* [in] - location of mouse (to determine if caption should be displayed)
|
|
|
|
// Output:
|
|
// HRESULT - various. S_OK if operation succeeded.
|
|
|
|
|
|
|
|
HRESULT CDeskMovr::MoveSelfToTarget(IHTMLElement *pielem, POINT * pptDoc)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::MoveSelfToTarget(pielem=%lx) %s, m_bstrTargetName=%ls", pielem, (pielem ? "We are GETTING focus." : "We are LOOSING focus."), GEN_DEBUGSTRW(m_bstrTargetName));
|
|
|
|
if (!pielem)
|
|
{
|
|
// The m_cSkipTimer variable is used to allow the skipping of timer ticks when determining
|
|
// if the mover should be dismissed. By doing this it gives the user more time and thus
|
|
// a better chance to manipulate the target if they are prone to drifting the mouse
|
|
// outside the target by accident.
|
|
|
|
// Check the m_cSkipTimer before dismissing the mover.
|
|
if (!m_cSkipTimer)
|
|
{
|
|
_ChangeCapture(FALSE);
|
|
if (m_pistyle)
|
|
hr = m_pistyle->put_visibility((BSTR)s_sstrHidden.wsz);
|
|
ATOMICRELEASE(m_pistyleTarget);
|
|
ATOMICRELEASE(m_pielemTarget);
|
|
m_iSrcTarget = -1;
|
|
}
|
|
else
|
|
{
|
|
m_cSkipTimer--;
|
|
hr = S_FALSE;
|
|
}
|
|
|
|
// These are actions we want to happen right away.
|
|
m_hcursor = CursorFromDragMode(dmNull);
|
|
if (m_hcursor != NULL)
|
|
SetCursor(m_hcursor);
|
|
}
|
|
|
|
// These are actions we want to happen after the Desktop Item
|
|
// looses focus.
|
|
if (hr != S_FALSE)
|
|
{
|
|
m_cyCaption = 0;
|
|
m_cxBorder = m_cxSMBorder;
|
|
m_cyBorder = m_cySMBorder;
|
|
m_CaptionState = 0;
|
|
m_dmTrack = dmNull;
|
|
}
|
|
|
|
if (pielem)
|
|
{
|
|
ASSERT(m_pielemTarget != pielem);
|
|
|
|
// exchange our new target ( if any ) for the old target, if any...
|
|
ATOMICRELEASE(m_pistyleTarget);
|
|
ATOMICRELEASE(m_pielemTarget);
|
|
|
|
hr = pielem->get_style(&m_pistyleTarget);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// We are gaining focus.
|
|
m_pielemTarget = pielem;
|
|
m_pielemTarget->AddRef();
|
|
|
|
EVAL(SUCCEEDED(InitAttributes(m_pielemTarget)));
|
|
|
|
if (!pptDoc)
|
|
TrackCaption(NULL);
|
|
TrackTarget(pptDoc);
|
|
// Set m_cSkipTimer so that we delay dismissing the mover...
|
|
m_cSkipTimer = GET_SKIP_COUNT;
|
|
if (!m_bWndLess && !m_hWnd)
|
|
{
|
|
// This is all a hack until trident fixes the UIDeactivate stuff, bug 243801
|
|
IOleInPlaceObject_InPlaceDeactivate();
|
|
InPlaceActivate(OLEIVERB_UIACTIVATE);
|
|
SetControlFocus(TRUE);
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
// CDeskMovrControl::MoveSelfAndTarget [instance method]
|
|
|
|
// Moves the control and it's target to a new location.
|
|
|
|
// Parameters:
|
|
// LONG [in] - x location, in document coord's to move to
|
|
// LONG [in] - y location, in document coord's to move to
|
|
|
|
// Output:
|
|
// HRESULT - various. S_OK if operation succeeded.
|
|
|
|
// Notes:
|
|
// We read back the target's location so that we stay consistent with
|
|
// any constraint's Trident might impose on our movement.
|
|
|
|
HRESULT CDeskMovr::MoveSelfAndTarget(LONG x, LONG y)
|
|
{
|
|
HRESULT hr;
|
|
|
|
m_top = y;
|
|
CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelTop(y - m_cyBorder - m_cyCaption)));
|
|
CLEANUP_ON_FAILURE((hr = m_pistyleTarget->put_pixelTop(y)));
|
|
// read it back to catch Trident constraint.
|
|
//CLEANUP_ON_FAILURE((hr = m_pielemTarget->get_docTop( &m_top )));
|
|
//CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelTop( m_top )));
|
|
|
|
m_left = x;
|
|
CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelLeft(x - m_cxBorder)));
|
|
CLEANUP_ON_FAILURE((hr = m_pistyleTarget->put_pixelLeft(x)));
|
|
// read it back to catch Trident constraint.
|
|
//CLEANUP_ON_FAILURE((hr = m_pielemTarget->get_docLeft( &m_left )));
|
|
//CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelLeft( m_left )));
|
|
|
|
// if ( !m_bWndLess )
|
|
if (EVAL(S_OK == GetParentWindow()))
|
|
::UpdateWindow(m_hwndParent);
|
|
|
|
CleanUp:
|
|
return hr;
|
|
}
|
|
|
|
BOOL CDeskMovr::FFindTargetElement(IHTMLElement *pielem, IHTMLElement **ppielem)
|
|
{
|
|
*ppielem = NULL;
|
|
|
|
if (pielem != NULL)
|
|
{
|
|
IDeskMovr *pidm = NULL;
|
|
|
|
// If it is over the mover return the current target, otherwise
|
|
// find out which component if any we are over.
|
|
if (m_pielemTarget != NULL &&
|
|
SUCCEEDED(pielem->QueryInterface(IID_IDeskMovr, (LPVOID*)&pidm)))
|
|
{
|
|
m_pielemTarget->AddRef();
|
|
*ppielem = m_pielemTarget;
|
|
ATOMICRELEASE(pidm);
|
|
}
|
|
else {
|
|
HRESULT hr;
|
|
IHTMLElement *pielem2 = pielem;
|
|
|
|
pielem2->AddRef();
|
|
|
|
do
|
|
{
|
|
VARIANT var;
|
|
|
|
VariantInit(&var);
|
|
|
|
if (SUCCEEDED(hr = pielem2->getAttribute((BSTR)s_sstrNameMember.wsz, TRUE, &var))) {
|
|
if (var.vt == VT_BSTR && var.bstrVal != NULL) {
|
|
if (StrCmpW(var.bstrVal, m_bstrTargetName) == 0)
|
|
hr = S_OK;
|
|
else
|
|
hr = S_FALSE;
|
|
}
|
|
else
|
|
hr = S_FALSE; // Try VariantChangeType?????
|
|
}
|
|
else
|
|
hr = S_FALSE; // not here, maybe in parent.
|
|
|
|
VariantClear(&var);
|
|
|
|
if (hr == S_OK) { // we found it
|
|
hr = pielem2->QueryInterface(IID_IHTMLElement, (LPVOID*)ppielem);
|
|
}
|
|
else if (hr == S_FALSE) { // not this one, climb up
|
|
IHTMLElement *pielemParent = NULL;
|
|
|
|
pielem2->get_parentElement(&pielemParent);
|
|
pielem2->Release(); // we're through at this level
|
|
pielem2 = pielemParent; // may be null, which just means we've reached the top.
|
|
}
|
|
|
|
} while (SUCCEEDED(hr) && *ppielem == NULL && pielem2 != NULL);
|
|
|
|
ATOMICRELEASE(pielem2);
|
|
}
|
|
}
|
|
|
|
return *ppielem != NULL;
|
|
}
|
|
|
|
|
|
// CDeskMovr::DragModeFromPoint [instance method]
|
|
|
|
// Moves the control and it's target to a new location.
|
|
|
|
// Parameters:
|
|
// POINT - point to test, in local coords
|
|
|
|
// Output:
|
|
// DragMode - drag mode associated with the point
|
|
|
|
// Notes:
|
|
// This is only a hit testing method. It does not alter state.
|
|
|
|
CDeskMovr::DragMode CDeskMovr::DragModeFromPoint(POINT pt)
|
|
{
|
|
enum DragMode dm = dmNull;
|
|
RECT rc;
|
|
|
|
if (PtInRect(&m_rectInner, pt))
|
|
{ // either no-hit, or on caption
|
|
if (PtInRect(&m_rectCaption, pt)) {
|
|
DragMode dmT;
|
|
|
|
for (dmT = dmMenu; dmT < dmMove; dmT = (DragMode)((int)dmT + 1)) {
|
|
if (GetCaptionButtonRect(dmT, &rc) && PtInRect(&rc, pt)) {
|
|
dm = dmT;
|
|
break;
|
|
}
|
|
}
|
|
if ((dmT == dmMove) && !CAPTION_ONLY)
|
|
dm = dmMove;
|
|
}
|
|
}
|
|
else {
|
|
if (PtInRect(&m_rectOuter, pt)) {
|
|
if (!CAPTION_ONLY)
|
|
{
|
|
// a resize border hit
|
|
if (pt.y <= m_sizeCorner.cy) {
|
|
// upper edge or corners
|
|
if (pt.x <= m_sizeCorner.cx)
|
|
dm = dmSizeWHTL;
|
|
else if (pt.x >= m_rectOuter.right - m_sizeCorner.cx)
|
|
dm = dmSizeWHTR;
|
|
else
|
|
dm = dmSizeTop;
|
|
}
|
|
else if (pt.y >= m_rectOuter.bottom - m_sizeCorner.cy) {
|
|
// bottom edge or corners
|
|
if (pt.x <= m_sizeCorner.cx)
|
|
dm = dmSizeWHBL;
|
|
else if (pt.x >= m_rectOuter.right - m_sizeCorner.cx)
|
|
dm = dmSizeWHBR;
|
|
else
|
|
dm = dmSizeBottom;
|
|
}
|
|
else {
|
|
// side edge hit
|
|
if (pt.x > m_rectInner.left)
|
|
dm = dmSizeRight;
|
|
else
|
|
dm = dmSizeLeft;
|
|
}
|
|
}
|
|
else {
|
|
if (m_cyCaption == 0)
|
|
dm = dmSizeLeft;
|
|
else
|
|
dm = dmNull;
|
|
}
|
|
}
|
|
//Check if this element can be sized in both the directions.
|
|
if (!m_fCanResizeX)
|
|
{
|
|
if ((dm != dmSizeTop) && (dm != dmSizeBottom))
|
|
dm = dmNull;
|
|
}
|
|
|
|
if (!m_fCanResizeY)
|
|
{
|
|
if ((dm != dmSizeLeft) && (dm != dmSizeRight))
|
|
dm = dmNull;
|
|
}
|
|
}
|
|
|
|
return dm;
|
|
}
|
|
|
|
// Align our member RECTs with the dimensions of the target element.
|
|
void CDeskMovr::SyncRectsToTarget(void)
|
|
{
|
|
// do the swelling thang
|
|
if ((m_dmTrack > dmMove) || m_cyCaption) {
|
|
m_cxBorder = GET_CXSIZE;
|
|
m_cyBorder = GET_CYSIZE;
|
|
}
|
|
else {
|
|
m_cxBorder = m_cxSMBorder;
|
|
m_cyBorder = m_cySMBorder;
|
|
}
|
|
|
|
m_rectOuter.top = m_rectOuter.left = 0;
|
|
|
|
if (CAPTION_ONLY)
|
|
{
|
|
if (m_cyCaption != 0)
|
|
{
|
|
// Displaying just caption
|
|
m_rectOuter.bottom = m_cyCaption + m_cyBorder;
|
|
m_rectOuter.right = m_width;
|
|
}
|
|
else {
|
|
// Displaying just left size border
|
|
m_rectOuter.bottom = m_height;
|
|
m_rectOuter.right = m_cxBorder;
|
|
}
|
|
}
|
|
else {
|
|
// Displaying caption and border
|
|
m_rectOuter.bottom = m_height + 2 * m_cyBorder + m_cyCaption;
|
|
m_rectOuter.right = m_width + 2 * m_cxBorder;
|
|
}
|
|
|
|
if (CAPTION_ONLY && m_cyCaption == 0)
|
|
{
|
|
// Displaying just left size border
|
|
SetRectEmpty(&m_rectInner);
|
|
SetRectEmpty(&m_rectCaption);
|
|
}
|
|
else {
|
|
// Displaying caption and possibly border
|
|
m_rectInner = m_rectOuter;
|
|
InflateRect(&m_rectInner, -m_cxBorder, -m_cyBorder);
|
|
|
|
m_rectCaption = m_rectInner;
|
|
m_rectCaption.bottom = m_cyBorder + m_cyCaption;
|
|
|
|
}
|
|
|
|
if (m_rectOuter.bottom > 2 * m_cyCaption)
|
|
m_sizeCorner.cy = GET_CYCAPTION;
|
|
else
|
|
m_sizeCorner.cy = m_rectOuter.bottom / 2;
|
|
|
|
if (m_rectOuter.right > 2 * m_cyCaption)
|
|
m_sizeCorner.cx = GET_CYCAPTION;
|
|
else
|
|
m_sizeCorner.cx = m_rectOuter.right / 2;
|
|
|
|
|
|
}
|
|
|
|
HCURSOR CDeskMovr::CursorFromDragMode(DragMode dm)
|
|
{
|
|
ASSERT(dm >= 0 && dm < cDragModes);
|
|
switch (dm) {
|
|
case dmNull:
|
|
case dmMenu:
|
|
case dmClose:
|
|
case dmMove:
|
|
case dmRestore:
|
|
case dmFullScreen:
|
|
case dmSplit:
|
|
default:
|
|
return LoadCursor(NULL, IDC_ARROW);
|
|
case dmSizeWHBR:
|
|
case dmSizeWHTL:
|
|
return LoadCursor(NULL, IDC_SIZENWSE);
|
|
case dmSizeWHTR:
|
|
case dmSizeWHBL:
|
|
return LoadCursor(NULL, IDC_SIZENESW);
|
|
case dmSizeTop:
|
|
case dmSizeBottom:
|
|
return LoadCursor(NULL, IDC_SIZENS);
|
|
case dmSizeLeft:
|
|
case dmSizeRight:
|
|
return LoadCursor(NULL, IDC_SIZEWE);
|
|
}
|
|
}
|
|
|
|
|
|
// CDeskMovr::SizeSelfAndTarget [instance method]
|
|
|
|
// Resizes our control and its target element.
|
|
|
|
// Parameters:
|
|
// LONG [in] - new width
|
|
// LONG [in] - new height
|
|
|
|
// Output:
|
|
// HRESULT - various. S_OK if operation succeeded.
|
|
|
|
// Notes:
|
|
// We read back the target's dimensions so that we stay consistent with
|
|
// any constraint's Trident might impose on our sizing.
|
|
|
|
HRESULT CDeskMovr::SizeSelfAndTarget(POINT ptDoc)
|
|
{
|
|
HRESULT hr;
|
|
int topOld = m_top;
|
|
int leftOld = m_left;
|
|
int heightOld = m_height;
|
|
int widthOld = m_width;
|
|
int cyCaption = GET_CYCAPTION;
|
|
|
|
switch (m_dmCur) {
|
|
case dmSizeWHBR:
|
|
m_width = (ptDoc.x + m_dx) - m_left;
|
|
m_height = (ptDoc.y + m_dy) - m_top;
|
|
break;
|
|
case dmSizeWHTL:
|
|
m_top = ptDoc.y + m_dy;
|
|
m_height += topOld - m_top;
|
|
m_left = ptDoc.x + m_dx;
|
|
m_width += leftOld - m_left;
|
|
break;
|
|
case dmSizeWHTR:
|
|
m_top = ptDoc.y + m_dy;
|
|
m_height += topOld - m_top;
|
|
m_width = (ptDoc.x + m_dx) - m_left;
|
|
break;
|
|
case dmSizeWHBL:
|
|
m_height = (ptDoc.y + m_dy) - m_top;
|
|
m_left = ptDoc.x + m_dx;
|
|
m_width += leftOld - m_left;
|
|
break;
|
|
case dmSizeTop:
|
|
m_top = ptDoc.y + m_dy;
|
|
m_height += topOld - m_top;
|
|
break;
|
|
case dmSizeBottom:
|
|
m_height = (ptDoc.y + m_dy) - m_top;
|
|
break;
|
|
case dmSizeLeft:
|
|
m_left = ptDoc.x + m_dx;
|
|
m_width += leftOld - m_left;
|
|
break;
|
|
case dmSizeRight:
|
|
m_width = (ptDoc.x + m_dx) - m_left;
|
|
break;
|
|
default:
|
|
ASSERT(FALSE);
|
|
return E_FAIL;
|
|
}
|
|
|
|
// limit shrinkage to keep the handle accessible
|
|
if (m_height < cyCaption) {
|
|
m_height = cyCaption;
|
|
if (m_top != topOld)
|
|
m_top = topOld + heightOld - m_height;
|
|
}
|
|
|
|
// limit shrinkage to keep the handle accessible
|
|
if (m_width < (4 * cyCaption)) {
|
|
m_width = 4 * cyCaption;
|
|
if (m_left != leftOld)
|
|
m_left = leftOld + widthOld - m_width;
|
|
}
|
|
|
|
SyncRectsToTarget();
|
|
|
|
if (m_top != topOld) {
|
|
CLEANUP_ON_FAILURE((hr = m_pistyleTarget->put_pixelTop(m_top)));
|
|
CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelTop(m_top - (m_cyBorder + m_cyCaption))));
|
|
}
|
|
|
|
if (m_left != leftOld) {
|
|
CLEANUP_ON_FAILURE((hr = m_pistyleTarget->put_pixelLeft(m_left)));
|
|
CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelLeft(m_left - (CAPTION_ONLY ? 0 : m_cxBorder))));
|
|
}
|
|
|
|
CLEANUP_ON_FAILURE((hr = m_pistyleTarget->put_pixelHeight(m_height)));
|
|
// read it back to catch Trident constraint.
|
|
//CLEANUP_ON_FAILURE((hr = m_pielemTarget->get_docHeight( &m_height )));
|
|
CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelHeight(m_rectOuter.bottom)));
|
|
|
|
CLEANUP_ON_FAILURE((hr = m_pistyleTarget->put_pixelWidth(m_width)));
|
|
// read it back to catch Trident constraint.
|
|
//CLEANUP_ON_FAILURE((hr = m_pielemTarget->get_docWidth( &m_width )));
|
|
CLEANUP_ON_FAILURE((hr = m_pistyle->put_pixelWidth(m_rectOuter.right)));
|
|
|
|
if (EVAL(S_OK == GetParentWindow()))
|
|
::UpdateWindow(m_hwndParent);
|
|
|
|
CleanUp:
|
|
return hr;
|
|
}
|
|
|
|
|
|
// IQuickActivate
|
|
HRESULT CDeskMovr::QuickActivate(QACONTAINER *pQACont, QACONTROL *pQACtrl)
|
|
{
|
|
HRESULT hr = IQuickActivate_QuickActivate(pQACont, pQACtrl);
|
|
|
|
if (pQACont)
|
|
{
|
|
ClearFlag(pQACtrl->dwViewStatus, VIEWSTATUS_OPAQUE);
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::_GetHTMLDoc(IOleClientSite * pocs, IHTMLDocument2 ** pphd2)
|
|
{
|
|
HRESULT hr;
|
|
IOleContainer * poc = NULL;
|
|
|
|
if (!EVAL(pocs) || !EVAL(pphd2))
|
|
return E_INVALIDARG;
|
|
|
|
*pphd2 = NULL;
|
|
hr = pocs->GetContainer(&poc);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
hr = poc->QueryInterface(IID_IHTMLDocument2, (LPVOID*)pphd2);
|
|
poc->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::_IsInElement(HWND hwndParent, POINT * ppt, IHTMLElement ** pphe)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
ASSERT(pphe);
|
|
|
|
*pphe = NULL;
|
|
if (!ppt || ::ScreenToClient(hwndParent, ppt))
|
|
{
|
|
IHTMLDocument2 * phd2;
|
|
|
|
ASSERT(m_spClientSite);
|
|
hr = _GetHTMLDoc(m_spClientSite, &phd2);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
if (ppt)
|
|
hr = phd2->elementFromPoint(ppt->x, ppt->y, pphe);
|
|
else
|
|
hr = phd2->get_activeElement(pphe);
|
|
|
|
if (!*pphe && SUCCEEDED(hr))
|
|
hr = E_FAIL; // Sometimes Trident returns S_FALSE on error.
|
|
|
|
phd2->Release();
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
HRESULT CDeskMovr::_EnumComponents(LPFNCOMPENUM lpfn, LPVOID lpvData, DWORD dwData)
|
|
{
|
|
HRESULT hr = E_FAIL;
|
|
IActiveDesktop * padt = NULL;
|
|
|
|
hr = CoCreateInstance(CLSID_ActiveDesktop, NULL, CLSCTX_INPROC, IID_IActiveDesktop, (LPVOID *)&padt);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
int nCount;
|
|
int nIndex;
|
|
|
|
hr = padt->GetDesktopItemCount(&nCount, 0);
|
|
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
COMPONENT comp;
|
|
|
|
for (nIndex = 0; nIndex < nCount; nIndex++)
|
|
{
|
|
comp.dwSize = sizeof(COMPONENT);
|
|
|
|
hr = padt->GetDesktopItem(nIndex, &comp, 0);
|
|
if (EVAL(SUCCEEDED(hr)))
|
|
{
|
|
if ((hr = lpfn(&comp, lpvData, dwData)) != S_OK)
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
padt->Release();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::_EnumElements(LPFNELEMENUM lpfn, LPVOID lpvData, DWORD dwData)
|
|
{
|
|
HRESULT hr;
|
|
IHTMLDocument2 * phd2;
|
|
|
|
ASSERT(m_spClientSite);
|
|
|
|
if (SUCCEEDED(hr = _GetHTMLDoc(m_spClientSite, &phd2)))
|
|
{
|
|
IHTMLElementCollection * pelems;
|
|
|
|
if (SUCCEEDED(hr = phd2->get_all(&pelems)))
|
|
{
|
|
VARIANT varIndex;
|
|
VARIANT varDummy;
|
|
IDispatch * pidisp;
|
|
|
|
VariantInit(&varDummy);
|
|
varIndex.vt = VT_I4;
|
|
varIndex.lVal = 0;
|
|
|
|
// Note: This loop terminates when trident returns SUCCESS - but with a NULL pidisp.
|
|
while (SUCCEEDED(hr = pelems->item(varIndex, varDummy, &pidisp)) && pidisp)
|
|
{
|
|
IHTMLElement * pielem;
|
|
|
|
if (SUCCEEDED(hr = pidisp->QueryInterface(IID_IHTMLElement, (LPVOID *)&pielem)))
|
|
{
|
|
hr = lpfn(pielem, lpvData, dwData);
|
|
pielem->Release();
|
|
}
|
|
|
|
pidisp->Release();
|
|
|
|
if (hr != S_OK)
|
|
break;
|
|
|
|
varIndex.lVal++;
|
|
}
|
|
|
|
pelems->Release();
|
|
}
|
|
phd2->Release();
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
HRESULT lpfnZOrderCB(COMPONENT * pcomp, LPVOID lpvData, DWORD dwData)
|
|
{
|
|
#define LPZORDERSLOT ((LONG *)lpvData)
|
|
|
|
if (dwData ? (pcomp->cpPos.izIndex > *LPZORDERSLOT) : (pcomp->cpPos.izIndex < *LPZORDERSLOT))
|
|
*LPZORDERSLOT = pcomp->cpPos.izIndex;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CDeskMovr::_GetZOrderSlot(LONG * plZOrderSlot, BOOL fTop)
|
|
{
|
|
HRESULT hr;
|
|
|
|
ASSERT(plZOrderSlot);
|
|
|
|
*plZOrderSlot = m_bWindowOnly ? 10000 : 5000;
|
|
|
|
hr = _EnumComponents(lpfnZOrderCB, (LPVOID)plZOrderSlot, (DWORD)fTop);
|
|
|
|
*plZOrderSlot += fTop ? 2 : -2; // Make sure we are above / below.
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
// PersistTargetPosition [helper function]
|
|
|
|
// Update the registry entries that are the persistence of the desktop HTML.
|
|
|
|
// Parameters:
|
|
// <none>
|
|
|
|
// Output:
|
|
// <none>
|
|
|
|
// Notes:
|
|
// If we fail, we do it quietly.
|
|
|
|
|
|
void PersistTargetPosition(IHTMLElement *pielem,
|
|
int left,
|
|
int top,
|
|
int width,
|
|
int height,
|
|
int zIndex,
|
|
BOOL fSaveState,
|
|
BOOL fSaveOriginal,
|
|
DWORD dwNewState)
|
|
{
|
|
// only do this persistence thing if we're in ( or completing ) an operation
|
|
TCHAR szID[MAX_ID_LENGTH];
|
|
BOOL fOK;
|
|
|
|
if (SUCCEEDED(GetHTMLElementStrMember(pielem, szID, SIZECHARS(szID), (BSTR)(s_sstrIDMember.wsz))))
|
|
{
|
|
bool bChangedPosition, bChangedSize;
|
|
COMPPOS compPos;
|
|
|
|
// 99/03/23 #266412 vtan: The user has moved the deskmovr to a new position
|
|
// make sure that it is within the work area of the display monitors.
|
|
// ValidateComponentPosition() will do this for us and tell us whether the
|
|
// the component got moved or resized.
|
|
|
|
compPos.dwSize = sizeof(compPos);
|
|
compPos.iLeft = left;
|
|
compPos.iTop = top;
|
|
compPos.dwWidth = width;
|
|
compPos.dwHeight = height;
|
|
|
|
ValidateComponentPosition(&compPos, dwNewState, COMP_TYPE_HTMLDOC, &bChangedPosition, &bChangedSize);
|
|
if (bChangedPosition || bChangedSize)
|
|
{
|
|
IHTMLStyle *pIStyle;
|
|
|
|
// If the component got moved or resized then tell the object model
|
|
// where the deskmovr is now.
|
|
|
|
left = compPos.iLeft;
|
|
top = compPos.iTop;
|
|
width = compPos.dwWidth;
|
|
height = compPos.dwHeight;
|
|
if (SUCCEEDED(pielem->get_style(&pIStyle)))
|
|
{
|
|
pIStyle->put_pixelLeft(left);
|
|
pIStyle->put_pixelTop(top);
|
|
pIStyle->put_pixelWidth(width);
|
|
pIStyle->put_pixelHeight(height);
|
|
pIStyle->Release();
|
|
}
|
|
}
|
|
fOK = UpdateDesktopPosition(szID, left, top, width, height, zIndex, fSaveState, fSaveOriginal, dwNewState);
|
|
}
|
|
|
|
TraceMsg(TF_CUSTOM2, "PersistTargetPosition(pielem=%s, <left=%d, top=%d, wid=%d, h=%d>)", szID, left, top, width, height);
|
|
|
|
}
|
|
|
|
void ObtainSavedStateForElem(IHTMLElement *pielem,
|
|
LPCOMPSTATEINFO pCompState, BOOL fRestoredState)
|
|
{
|
|
// only do this persistence thing if we're in ( or completing ) an operation
|
|
TCHAR szID[MAX_ID_LENGTH];
|
|
BOOL fOK;
|
|
|
|
if (SUCCEEDED(GetHTMLElementStrMember(pielem, szID, SIZECHARS(szID), (BSTR)(s_sstrIDMember.wsz))))
|
|
fOK = GetSavedStateInfo(szID, pCompState, fRestoredState);
|
|
|
|
TraceMsg(TF_CUSTOM2, "ObtainSavedStateForElem(pielem=%s, <left=%d, top=%d, wid=%d, h=%d>)", szID, pCompState->iLeft, pCompState->iTop, pCompState->dwWidth, pCompState->dwHeight);
|
|
|
|
}
|
|
|
|
// IOleObject
|
|
HRESULT CDeskMovr::GetMiscStatus(DWORD dwAspect, DWORD *pdwStatus)
|
|
{
|
|
if (dwAspect == DVASPECT_CONTENT)
|
|
{
|
|
*pdwStatus = OLEMISMOVR;
|
|
return S_OK;
|
|
}
|
|
else
|
|
{
|
|
return DV_E_DVASPECT;
|
|
}
|
|
|
|
// dead code
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::SetClientSite(IOleClientSite * pClientSite)
|
|
{
|
|
if (!pClientSite)
|
|
DeactivateMovr(FALSE);
|
|
|
|
return CComControlBase::IOleObject_SetClientSite(pClientSite);
|
|
}
|
|
|
|
void HandleRestore(IHTMLElement * pielem)
|
|
{
|
|
VARIANT varZ;
|
|
COMPSTATEINFO csiRestore;
|
|
IHTMLStyle * pistyle;
|
|
|
|
if (SUCCEEDED(pielem->get_style(&pistyle)))
|
|
{
|
|
csiRestore.dwSize = sizeof(csiRestore);
|
|
|
|
ObtainSavedStateForElem(pielem, &csiRestore, TRUE); // TRUE => Get restored state!
|
|
|
|
pistyle->put_pixelLeft(csiRestore.iLeft);
|
|
pistyle->put_pixelTop(csiRestore.iTop);
|
|
pistyle->put_pixelWidth(csiRestore.dwWidth);
|
|
pistyle->put_pixelHeight(csiRestore.dwHeight);
|
|
|
|
varZ.vt = VT_I4;
|
|
pistyle->get_zIndex(&varZ);
|
|
|
|
PersistTargetPosition(pielem, csiRestore.iLeft, csiRestore.iTop, csiRestore.dwWidth, csiRestore.dwHeight, varZ.lVal, FALSE, FALSE, IS_NORMAL);
|
|
pistyle->Release();
|
|
}
|
|
}
|
|
|
|
HRESULT lpfnRestoreCB(IHTMLElement * pielem, LPVOID lpvData, DWORD dwData)
|
|
{
|
|
HRESULT hres = S_OK;
|
|
TCHAR szID[MAX_ID_LENGTH];
|
|
|
|
if (SUCCEEDED(GetHTMLElementStrMember(pielem, szID, SIZECHARS(szID), (BSTR)(s_sstrIDMember.wsz))))
|
|
{
|
|
DWORD dwState = GetCurrentState(szID);
|
|
|
|
// Since there is only one in this state we can stop the enumeration if we
|
|
// find a fullscreen/split item on this work area.
|
|
if (dwState & (IS_FULLSCREEN | IS_SPLIT)) {
|
|
POINT pt;
|
|
if (SUCCEEDED(CSSOM_TopLeft(pielem, &pt)) && PtInRect((CONST RECT *)lpvData, pt))
|
|
{
|
|
HandleRestore(pielem);
|
|
hres = S_FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hres;
|
|
}
|
|
|
|
|
|
HRESULT CDeskMovr::_HandleZoom(LONG lCommand)
|
|
{
|
|
LONG x, y, cx, cy, zIndex;
|
|
VARIANT varZ;
|
|
DWORD dwOldItemState = m_ItemState, dwNewItemState;
|
|
IHTMLStyle * pistyleTarget = m_pistyleTarget;
|
|
IHTMLElement * pielemTarget = m_pielemTarget;
|
|
|
|
// Paranoia
|
|
if (!pistyleTarget || !pielemTarget)
|
|
{
|
|
ASSERT(FALSE);
|
|
return E_FAIL;
|
|
}
|
|
|
|
// Hold on to these guys during this call, they could go away when we yield
|
|
// like during the animation call below.
|
|
pistyleTarget->AddRef();
|
|
pielemTarget->AddRef();
|
|
|
|
if (lCommand == IDM_DCCM_RESTORE)
|
|
{
|
|
COMPSTATEINFO csi;
|
|
csi.dwSize = sizeof(csi);
|
|
|
|
// The "Restore" command toggles with the "Reset Original Size" command.
|
|
// Make sure we get the correct Restore or Reset position for the element.
|
|
ObtainSavedStateForElem(pielemTarget, &csi, !ISNORMAL);
|
|
|
|
if (ISNORMAL)
|
|
{
|
|
// This is the split case, dont move the item just resize it.
|
|
x = m_left;
|
|
y = m_top;
|
|
}
|
|
else
|
|
{
|
|
|
|
// 98/07/27 vtan #176721: The following checks restoration of a component
|
|
// position from zoomed to user-specified position. If the component
|
|
// is placed at the default position then it is positioned now using
|
|
// the standard positioning code.
|
|
|
|
if ((csi.iLeft == COMPONENT_DEFAULT_LEFT) &&
|
|
(csi.iTop == COMPONENT_DEFAULT_TOP) &&
|
|
(csi.dwWidth == COMPONENT_DEFAULT_WIDTH) &&
|
|
(csi.dwHeight == COMPONENT_DEFAULT_HEIGHT))
|
|
{
|
|
COMPPOS compPos;
|
|
|
|
GetNextComponentPosition(&compPos);
|
|
IncrementComponentsPositioned();
|
|
csi.iLeft = compPos.iLeft;
|
|
csi.iTop = compPos.iTop;
|
|
csi.dwWidth = compPos.dwWidth;
|
|
csi.dwHeight = compPos.dwHeight;
|
|
}
|
|
// Restore case, go ahead and move it.
|
|
x = csi.iLeft;
|
|
y = csi.iTop;
|
|
}
|
|
|
|
cx = csi.dwWidth;
|
|
cy = csi.dwHeight;
|
|
m_ItemState = (m_ItemState & ~IS_VALIDSIZESTATEBITS) | IS_NORMAL;
|
|
dwNewItemState = m_ItemState;
|
|
|
|
m_zIndexTop += 2;
|
|
zIndex = m_zIndexTop;
|
|
}
|
|
else
|
|
{
|
|
RECT rcZoom, rcWork;
|
|
|
|
GetZoomRect(lCommand == IDM_DCCM_FULLSCREEN, TRUE, m_left, m_top, m_width, m_height, &rcZoom, &rcWork);
|
|
|
|
x = rcZoom.left;
|
|
y = rcZoom.top;
|
|
cx = rcZoom.right - rcZoom.left;
|
|
cy = rcZoom.bottom - rcZoom.top;
|
|
|
|
if (lCommand == IDM_DCCM_FULLSCREEN)
|
|
{
|
|
m_ItemState = (m_ItemState & ~IS_VALIDSIZESTATEBITS) | IS_FULLSCREEN;
|
|
dwNewItemState = m_ItemState;
|
|
}
|
|
else
|
|
{
|
|
m_ItemState = (m_ItemState & ~IS_VALIDSIZESTATEBITS) | IS_SPLIT;
|
|
dwNewItemState = m_ItemState;
|
|
}
|
|
|
|
// We currently only allow 1 component to be either split or full screen per monitor (WorkArea), so
|
|
// restore any other component that is currently in this state.
|
|
_EnumElements(lpfnRestoreCB, (LPVOID)&rcWork, 0);
|
|
|
|
m_zIndexBottom -= 2;
|
|
zIndex = m_zIndexBottom;
|
|
}
|
|
|
|
// We want to do the animation call before we start moving the target, it looks better
|
|
// that way.
|
|
AnimateComponent(m_hwndParent, m_left, m_top, m_width, m_height, x, y, cx, cy);
|
|
|
|
pistyleTarget->put_pixelLeft(x);
|
|
pistyleTarget->put_pixelTop(y);
|
|
pistyleTarget->put_pixelWidth(cx);
|
|
pistyleTarget->put_pixelHeight(cy);
|
|
|
|
varZ.vt = VT_I4;
|
|
varZ.lVal = zIndex;
|
|
pistyleTarget->put_zIndex(varZ);
|
|
|
|
PersistTargetPosition(pielemTarget, x, y, cx, cy, zIndex,
|
|
(BOOL)((dwOldItemState & IS_NORMAL) && !(dwNewItemState & IS_NORMAL)),
|
|
FALSE, dwNewItemState);
|
|
|
|
pistyleTarget->Release();
|
|
pielemTarget->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
/**\
|
|
FUNCTION: CDeskMovr::_DisplayContextMenu
|
|
|
|
PARAMETERS:
|
|
x,y - Coordinates relative to the desktop window.
|
|
\**/
|
|
HRESULT CDeskMovr::_DisplayContextMenu()
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HMENU hmenuContext = LoadMenuPopup(MENU_DESKCOMP_CONTEXTMENU);
|
|
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::DisplayContextMenu(), m_bstrTargetName=%ls", GEN_DEBUGSTRW(m_bstrTargetName));
|
|
if (hmenuContext)
|
|
{
|
|
int nSelection;
|
|
BOOL fSubscribe = FALSE;
|
|
BOOL fRemoveSubscribe = FALSE;
|
|
TCHAR szName[MAX_URL_STRING];
|
|
POINT point;
|
|
|
|
if (CAPTION_ONLY)
|
|
{
|
|
point.x = m_left + m_cxBorder;
|
|
point.y = m_top + (m_cyCaption + m_cyBorder) - 4 * m_cySMBorder;
|
|
}
|
|
else {
|
|
point.x = m_left - m_cxSMBorder;
|
|
point.y = m_top - 4 * m_cySMBorder;
|
|
}
|
|
|
|
::ClientToScreen(m_hwndParent, &point);
|
|
|
|
// BUGBUG: This calculation needs to be revisited. The reason it's so
|
|
// ugle and HACKy is because to look good, we want the context menu
|
|
// to appear on top of the 3-D edge below the triangle.
|
|
|
|
|
|
if (SUCCEEDED(GetHTMLElementStrMember(m_pielemTarget, szName, SIZECHARS(szName), (BSTR)(s_sstrSubSRCMember.wsz))))
|
|
{
|
|
int nScheme = GetUrlScheme(szName);
|
|
|
|
if ((URL_SCHEME_FILE == nScheme) || (URL_SCHEME_INVALID == nScheme))
|
|
fRemoveSubscribe = TRUE;
|
|
}
|
|
|
|
// check to see if we need to turn some things off or on
|
|
// Mainly because we are disabling features Admins don't want users to have.
|
|
|
|
hr = IElemCheckForExistingSubscription(m_pielemTarget);
|
|
if (fRemoveSubscribe || FAILED(hr)) // This object/thing cannot be subscribed to. (Channel Changer, Orenge Blob).
|
|
{
|
|
DeleteMenu(hmenuContext, IDM_DCCM_OFFLINE, MF_BYCOMMAND);
|
|
DeleteMenu(hmenuContext, IDM_DCCM_SYNCHRONIZE, MF_BYCOMMAND);
|
|
DeleteMenu(hmenuContext, IDM_DCCM_PROPERTIES, MF_BYCOMMAND);
|
|
|
|
// Is the top item in the list a separator?
|
|
if (-1 == GetMenuItemID(hmenuContext, 0))
|
|
{
|
|
// Yes, it is, so remove it.
|
|
DeleteMenu(hmenuContext, 0, MF_BYPOSITION);
|
|
}
|
|
}
|
|
else if (S_FALSE == hr) // Not subscribed
|
|
{
|
|
DeleteMenu(hmenuContext, IDM_DCCM_SYNCHRONIZE, MF_BYCOMMAND);
|
|
DeleteMenu(hmenuContext, IDM_DCCM_PROPERTIES, MF_BYCOMMAND);
|
|
fSubscribe = TRUE;
|
|
}
|
|
else if (S_OK == hr)
|
|
{
|
|
if (SHRestricted2(REST_NoManualUpdates, NULL, 0))
|
|
DeleteMenu(hmenuContext, IDM_DCCM_SYNCHRONIZE, MF_BYCOMMAND);
|
|
if (SHRestricted(REST_NOEDITDESKCOMP))
|
|
DeleteMenu(hmenuContext, IDM_DCCM_PROPERTIES, MF_BYCOMMAND);
|
|
|
|
CheckMenuItem(hmenuContext, IDM_DCCM_OFFLINE, MF_BYCOMMAND | MF_CHECKED);
|
|
}
|
|
|
|
if (SHRestricted(REST_NOCLOSEDESKCOMP))
|
|
EnableMenuItem(hmenuContext, IDM_DCCM_CLOSE, MF_BYCOMMAND | MF_GRAYED);
|
|
|
|
if (ISNORMAL)
|
|
{
|
|
COMPSTATEINFO CompState;
|
|
LoadString(HINST_THISDLL, IDS_MENU_RESET, szName, ARRAYSIZE(szName));
|
|
ModifyMenu(hmenuContext, IDM_DCCM_RESTORE, MF_BYCOMMAND | MF_STRING, IDM_DCCM_RESTORE, szName);
|
|
ObtainSavedStateForElem(m_pielemTarget, &CompState, FALSE);
|
|
if ((CompState.dwWidth == COMPONENT_DEFAULT_WIDTH && CompState.dwHeight == COMPONENT_DEFAULT_HEIGHT) ||
|
|
(CompState.dwWidth == (DWORD)m_width && CompState.dwHeight == (DWORD)m_height))
|
|
EnableMenuItem(hmenuContext, IDM_DCCM_RESTORE, MF_BYCOMMAND | MF_GRAYED);
|
|
}
|
|
if (ISSPLIT || !m_fCanResizeX || !m_fCanResizeY)
|
|
EnableMenuItem(hmenuContext, IDM_DCCM_SPLIT, MF_BYCOMMAND | MF_GRAYED);
|
|
if (ISFULLSCREEN || !m_fCanResizeX || !m_fCanResizeY)
|
|
EnableMenuItem(hmenuContext, IDM_DCCM_FULLSCREEN, MF_BYCOMMAND | MF_GRAYED);
|
|
|
|
g_fIgnoreTimers |= IGNORE_CONTEXTMENU_UP;
|
|
|
|
nSelection = TrackPopupMenu(hmenuContext, TPM_LEFTBUTTON | TPM_RIGHTBUTTON | TPM_NONOTIFY | TPM_RETURNCMD, point.x, point.y, 0, m_hwndParent, NULL);
|
|
|
|
DestroyMenu(hmenuContext);
|
|
|
|
m_CaptionState &= ~CS_MENUPUSHED;
|
|
UpdateCaption(DMDC_MENU);
|
|
|
|
switch (nSelection)
|
|
{
|
|
case IDM_DCCM_OFFLINE:
|
|
if (fSubscribe)
|
|
hr = IElemSubscribeDialog(m_pielemTarget, m_hWnd);
|
|
else
|
|
hr = IElemUnsubscribe(m_pielemTarget);
|
|
break;
|
|
|
|
case IDM_DCCM_SYNCHRONIZE:
|
|
hr = IElemUpdate(m_pielemTarget);
|
|
break;
|
|
|
|
case IDM_DCCM_PROPERTIES: // Subscriptions Dialog (Don't let the name fool you)
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::_DisplayContextMenu() IDM_DCCM_PROPERTIES m_bstrTargetName=%ls.", GEN_DEBUGSTRW(m_bstrTargetName));
|
|
hr = IElemGetSubscriptionsDialog(m_pielemTarget, NULL);
|
|
break;
|
|
|
|
case IDM_DCCM_CUSTOMIZE: // Show Display Control Panel set to Components Sheet
|
|
LoadString(HINST_THISDLL, IDS_COMPSETTINGS, szName, ARRAYSIZE(szName));
|
|
SHRunControlPanel(szName, NULL);
|
|
hr = S_OK;
|
|
break;
|
|
|
|
case IDM_DCCM_CLOSE:
|
|
ASSERT(!SHRestricted(REST_NOCLOSEDESKCOMP)); // We should never be able to get here.
|
|
|
|
TraceMsg(TF_CUSTOM2, "CDeskMovr::_DisplayContextMenu() IDM_DCCM_CLOSE m_bstrTargetName=%ls", GEN_DEBUGSTRW(m_bstrTargetName));
|
|
// AnimateToTray(m_hwndParent, m_left, m_top, m_width, m_height);
|
|
hr = IElemCloseDesktopComp(m_pielemTarget);
|
|
break;
|
|
|
|
case IDM_DCCM_RESTORE:
|
|
case IDM_DCCM_FULLSCREEN:
|
|
case IDM_DCCM_SPLIT:
|
|
hr = _HandleZoom(nSelection);
|
|
break;
|
|
|
|
case IDM_DCCM_OPEN:
|
|
{
|
|
BOOL fShowFrame = (GetKeyState(VK_SHIFT) < 0) ? !(m_fCanResizeX && m_fCanResizeY) : (m_fCanResizeX && m_fCanResizeY);
|
|
hr = IElemOpenInNewWindow(m_pielemTarget, m_spClientSite, fShowFrame, m_width, m_height);
|
|
}
|
|
break;
|
|
|
|
}
|
|
|
|
g_fIgnoreTimers &= ~IGNORE_CONTEXTMENU_UP;
|
|
|
|
if (nSelection)
|
|
DismissSelfNow();
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
void CDeskMovr::_MapPoints(int * px, int * py)
|
|
{
|
|
if (m_bWndLess)
|
|
{
|
|
*px -= m_left - (CAPTION_ONLY ? 0 : m_cxBorder);
|
|
*py -= (CAPTION_ONLY ? 0 : (m_top - (m_cyBorder + m_cyCaption)));
|
|
}
|
|
}
|
|
|
|
void CDeskMovr::_ChangeCapture(BOOL fSet)
|
|
{
|
|
if (m_fCaptured != fSet)
|
|
{
|
|
m_fCaptured = fSet;
|
|
if (fSet)
|
|
{
|
|
ASSERT(m_spInPlaceSite);
|
|
if (m_bWndLess && m_spInPlaceSite)
|
|
{
|
|
m_fCaptured = SUCCEEDED(m_spInPlaceSite->SetCapture(TRUE));
|
|
}
|
|
else
|
|
{
|
|
::SetCapture(m_hWnd);
|
|
m_fCaptured = (GetCapture() == m_hWnd);
|
|
}
|
|
if (m_fCaptured)
|
|
g_fIgnoreTimers |= IGNORE_CAPTURE_SET;
|
|
}
|
|
else
|
|
{
|
|
ASSERT(m_spInPlaceSite);
|
|
if (m_bWndLess && m_spInPlaceSite)
|
|
{
|
|
m_spInPlaceSite->SetCapture(FALSE);
|
|
}
|
|
else
|
|
{
|
|
ReleaseCapture();
|
|
}
|
|
|
|
g_fIgnoreTimers &= ~IGNORE_CAPTURE_SET;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Called from our keyboard hook so that we can implement keyboard invocation and dismissal
|
|
// of the deskmovr.
|
|
void CDeskMovr::OnKeyboardHook(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
IHTMLElement * pielem;
|
|
HWND hwndFocus = GetFocus();
|
|
|
|
if (!(g_fIgnoreTimers & IGNORE_CONTEXTMENU_UP) && SUCCEEDED(GetParentWindow()) && ((hwndFocus == m_hwndParent) || ::IsChild(m_hwndParent, hwndFocus)))
|
|
{
|
|
switch (wParam) {
|
|
case VK_MENU:
|
|
if (!m_pielemTarget && SUCCEEDED(SmartActivateMovr(ERROR_SUCCESS)) && SUCCEEDED(_IsInElement(NULL, NULL, &pielem)))
|
|
{
|
|
BOOL fDummy;
|
|
_TrackElement(NULL, pielem, &fDummy);
|
|
pielem->Release();
|
|
}
|
|
break;
|
|
case VK_ESCAPE:
|
|
case VK_TAB:
|
|
if ((lParam >= 0) && m_pielemTarget) // If key down, dismiss
|
|
DismissSelfNow();
|
|
break;
|
|
case VK_SPACE:
|
|
if (m_pielemTarget && (GET_CYCAPTION == m_cyCaption) && (HIWORD(lParam) & KF_ALTDOWN))
|
|
{
|
|
HandleNonMoveSize(dmMenu);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
STDAPI CDeskMovr_CreateInstance(IUnknown *pUnkOuter, REFIID riid, void **ppunk)
|
|
{
|
|
return CComCreator< CComPolyObject< CDeskMovr > >::CreateInstance((LPVOID)pUnkOuter, IID_IUnknown, (LPVOID*)ppunk);
|
|
}
|