gecko-dev/cmd/winfe/cxwin.cpp
1998-03-28 02:44:41 +00:00

7192 lines
228 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
*
* The contents of this file are subject to the Netscape Public License
* Version 1.0 (the "NPL"); you may not use this file except in
* compliance with the NPL. You may obtain a copy of the NPL at
* http://www.mozilla.org/NPL/
*
* Software distributed under the NPL is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
* for the specific language governing rights and limitations under the
* NPL.
*
* The Initial Developer of this code under the NPL is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
* Reserved.
*/
#include "stdafx.h"
#include "cxwin.h"
#include "cntritem.h"
#include "medit.h"
#include "button.h"
#include "fmbutton.h"
#include "gridedge.h"
#include "cxsave.h"
#include "netsvw.h"
#include "shcut.h"
#include "mainfrm.h"
#include "dialog.h"
#include "cxprint.h"
#include "findrepl.h"
#include "feembed.h"
#include "libevent.h"
#include "np.h"
#include "nppg.h"
#include "nppriv.h"
#include "winclose.h"
#include "tooltip.h"
#include "slavewnd.h"
#include "intl_csi.h"
#include "abdefn.h"
#include "feimage.h"
#include "edt.h"
extern char * EDT_NEW_DOC_URL;
extern char * EDT_NEW_DOC_NAME;
#define ED_SIZE_FEEDBACK_BORDER 3
#define EDT_IS_SIZING ( EDT_IS_EDITOR(GetContext()) && EDT_IsSizing(GetContext()) )
#ifdef JAVA
#include "np.h"
#include "java.h"
#include "prlog.h"
#ifdef DDRAW
static DDSURFACEDESC ddsd;
#endif
extern "C" {
#ifndef NSPR20
PR_LOG_DEFINE(APPLET);
#else
extern PRLogModuleInfo *APPLET;
#endif
/*
** API for querying the Navigator's Color Palette from the "outside"...
**
** This is used by AWT, when it is first started, to clone the navigator's
** color palette.
*/
PR_PUBLIC_API(HPALETTE) GET_APPLICATION_PALETTE(void)
{
HPALETTE hPal = (HPALETTE)NULL;
CWinCX *pActiveContext;
CGenericFrame *pFrame;
// XXX This is busted/need to move palette in frame -- Hokay?
pFrame = ((CNetscapeApp *)AfxGetApp())->m_pFrameList;
if( pFrame ) {
pActiveContext = pFrame->GetActiveWinContext();
if( pActiveContext ) {
hPal = pActiveContext->GetPalette();
}
}
return hPal;
}
};
#endif /* JAVA */
// older versions of MFC don't have this #define
#ifndef DEFAULT_GUI_FONT
#define DEFAULT_GUI_FONT ANSI_FIXED_FONT
#endif
#define NOT_A_DIALOG(wincx) \
((wincx) ? \
(wincx)->GetFrame() ? \
(wincx)->GetFrame()->GetMainContext() ? \
(wincx)->GetFrame()->GetMainContext()->GetContext() ? \
(wincx)->GetFrame()->GetMainContext()->GetContext()->type != MWContextDialog \
: TRUE : TRUE : TRUE : TRUE)
// An empty frame API so that this code will work without the presence of a frame window.
CNullFrame *CWinCX::m_pNullFrame = NULL;
void *CWinCX::m_pExitCookie = NULL;
void wincx_exit(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
// ExitInstance, clean up if possible.
if(CWinCX::m_pNullFrame) {
delete CWinCX::m_pNullFrame;
CWinCX::m_pNullFrame = NULL;
}
if(CWinCX::m_pExitCookie) {
slavewnd.UnRegister(CWinCX::m_pExitCookie);
CWinCX::m_pExitCookie = NULL;
}
}
CWinCX::CWinCX(CGenericDoc *pDocument, CFrameGlue *pFrame, CGenericView *pView, MWContextType mwType, ContextType cxType) : CPaneCX(pView ? pView->m_hWnd : NULL, FALSE) {
// Purpose: Construct a window context
// Arguments: pDocument The document, if NULL, one is created in CDCCX.
// It should be noted that this has to be a CNetscapeDoc
// currently, since it handles certain Ole aspects in
// the Netscape way.
// pFrame The frame which owns this view. Can be NULL (turns off
// any frame access).
// hView The view which this context interacts with. Can not be NULL.
// Returns: none
// Comments: Basically set the context type.
// Revision History:
// 06-25-95 created GAB
//
// Set the context type.
m_cxType = cxType;
GetContext()->type = mwType; // by default only.
m_pGenView = pView;
// No previous mouse event in this context.
m_LastMouseEvent = m_None;
m_bScrollingTimerSet = FALSE;
m_bMouseMoveTimerSet = FALSE;
// We start off thinking we have a border when a frame cell.
m_bHasBorder = TRUE;
m_MM = MM_TEXT;
// And the frame.
m_pFrame = pFrame;
pLastToolTipImg = 0;
m_pLastToolTipAnchor = NULL;
// Set up a callback in exit instance.
if(NULL == m_pExitCookie) {
m_pExitCookie = slavewnd.Register(SLAVE_EXITINSTANCE, wincx_exit);
}
// If there's no NULL frame yet, create one.
if(NULL == m_pNullFrame) {
m_pNullFrame = new CNullFrame;
}
// If there's no frame, use the Null Frame.
if(m_pFrame == NULL) {
TRACE("Using the NULL frame\n");
m_pFrame = m_pNullFrame;
}
// And the document.
m_pDocument = pDocument;
#ifdef DDRAW
m_ScrollWindow = FALSE;
m_physicWinRect.Empty();
#endif
// We're not active.
m_bActiveState = FALSE;
// We're not laying out.
m_bIsLayingOut = FALSE;
// We've not highlighted an anchor.
m_pLastArmedAnchor = NULL;
// No old progress;
m_lOldPercent = 0;
// not doing a drag/drop operation
m_bDragging = FALSE;
// No selected embedded item.
m_pSelected = NULL;
m_bSelectingCells = FALSE;
m_bInPopupMenu = FALSE;
// Inform the view of its new context.
if(m_pGenView) {
m_pGenView->SetContext(this);
}
/*__EDITOR__*/
m_bMouseInSelection = FALSE;
// We've not over an image.
m_pLastImageObject = NULL;
m_crWindowRect = CRect(0, 0, 0, 0);
// Mouse is up not down.
m_bLBDown = FALSE;
m_bLBUp = TRUE;
m_uMouseFlags = 0;
m_cpMMove.x = m_cpMMove.y = 0;
m_ToolTip = 0;
// size is not chrome specified yet.
m_bSizeIsChrome = FALSE;
// Clear those old elements that we will track in the
// loaded page (see FireMouseOverEvent....)
m_pLastOverAnchorData = NULL;
m_pLastOverElement = NULL;
m_pStartSelectionCell = NULL;
m_bLastOverTextSet = FALSE;
//#ifndef NO_TAB_NAVIGATION
m_lastTabFocus.pElement = NULL;
m_lastTabFocus.mapAreaIndex = 0; // 0 means no focus, start with index 1.
m_lastTabFocus.pAnchor = NULL;
m_isReEntry_setLastTabFocusElement = 0; // to provent re-entry
//#endif /* NO_TAB_NAVIGATION */
imageToolTip = NULL;
}
CWinCX::~CWinCX() {
// Purpose: Destroy a window context.
// Arguments: none
// Returns: none
// Comments: Does context instance specific cleanup.
// Revision History:
// 06-25-95 created GABby
//
MWContext *pContext = GetContext();
if(GetFrame() != NULL) {
GetFrame()->ClearContext(this);
}
if (m_ToolTip)
delete m_ToolTip;
XP_ListDestroy (imageToolTip);
// Netcaster going away
if (GetContext() == theApp.m_pNetcasterWindow)
theApp.m_pNetcasterWindow = NULL ;
}
#ifdef DDRAW
void CWinCX::CalcWinPos()
{
RECT tempRect1, tempRect2;
::GetWindowRect(GetPane(), &tempRect1);
::GetClientRect(GetPane(), &tempRect2);
int wWidth, cWidth;
int wHeight, cHeight;
int width, height;
wWidth = tempRect1.right - tempRect1.left;
cWidth = tempRect2.right - tempRect2.left;
width = wWidth - cWidth;
if (IsVScrollBarOn())
width -= sysInfo.m_iScrollWidth;
m_physicWinRect.left = tempRect1.left + (width / 2);
m_physicWinRect.right = tempRect1.right - (width / 2);
wHeight = tempRect1.bottom - tempRect1.top;
cHeight = tempRect2.bottom - tempRect2.top;
height = wHeight - cHeight;
if (IsHScrollBarOn())
height -= sysInfo.m_iScrollHeight;
m_physicWinRect.top = tempRect1.top + (height / 2);
m_physicWinRect.bottom = tempRect1.bottom - (width / 2);
}
#endif
void CWinCX::Initialize(BOOL bOwnDC, RECT *pRect, BOOL bInitialPalette, BOOL bNewMemDC) {
// Purpose: Initialize properties of this context.
// Arguments: void
// Returns: void
// Comments: Basically set up the iWidth and iHeight of the context.
// To be called after initial construction.
// Revision History:
// 06-25-95 created GABby
//
CPaneCX::Initialize(bOwnDC, pRect, bInitialPalette, bNewMemDC);
#ifdef DDRAW
m_lpDDSPrimary = NULL;
m_lpDDSBack = NULL;
mg_pPal = NULL;
m_pClipper = NULL;
m_pOffScreenClipper = NULL;
m_offScreenDC = 0;
m_lpDD = NULL;
/*
* create the main DirectDraw object
*/
if (DirectDrawCreate( NULL, &m_lpDD, NULL ) != DD_OK) {
m_lpDD->Release();
m_lpDD = NULL;
}
// initialize DirectDraw stuff.
if (m_lpDD && m_lpDD->SetCooperativeLevel( GetPane(), DDSCL_NORMAL )!= DD_OK) {
ReleaseDrawSurface();
}
if (m_lpDD) {
SetUseDibPalColors(FALSE);
// DDSURFACEDESC ddsd;
DDCAPS ddscaps, ddshel;
ZeroMemory((void*)&ddscaps, sizeof(DDCAPS));
// Create the primary surface with 1 back buffer
ddscaps.dwSize = sizeof( DDCAPS );
ZeroMemory((void*)&ddshel, sizeof(DDCAPS));
// Create the primary surface with 1 back buffer
ddshel.dwSize = sizeof( DDCAPS );
m_lpDD->GetCaps(&ddscaps, &ddshel);
ZeroMemory((void*)&ddsd, sizeof(DDSURFACEDESC));
// Create the primary surface with 1 back buffer
ddsd.dwSize = sizeof( ddsd );
ddsd.dwFlags = DDSD_CAPS;
ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE;
if (m_lpDD->CreateSurface( &ddsd, &m_lpDDSPrimary, NULL ) != DD_OK) {
ReleaseDrawSurface();
}
else { // create Offscreen draw surface.
CreateAndLockOffscreenSurface(*pRect);
// now create the palette for direct draw surface.
if (m_lpDDSPrimary) {
ZeroMemory(&m_surfDesc, sizeof(ddsd));
m_surfDesc.dwSize = sizeof(ddsd);
m_surfDesc.dwFlags = DDSD_PIXELFORMAT;
m_lpDDSBack->GetSurfaceDesc(&m_surfDesc);
DDPIXELFORMAT pFormat;
m_lpDDSPrimary->GetPixelFormat(&pFormat);
// If this surface supports a palette, then do it
if (m_surfDesc.ddpfPixelFormat.dwFlags & DDPF_PALETTEINDEXED8) {
PALETTEENTRY palEntry[256];
::GetPaletteEntries(GetPalette(), 0, 255, palEntry);
if (m_lpDD->CreatePalette(DDPCAPS_8BIT, palEntry, &mg_pPal, NULL) != DD_OK) {
ReleaseDrawSurface();
}
else {
if (m_lpDDSPrimary->SetPalette(mg_pPal )!= DD_OK) {
ReleaseDrawSurface();
}
else {
m_lpDDSBack->SetPalette(mg_pPal );
}
}
}
// Use a clipper object for clipping when in windowed mode
if (m_lpDD->CreateClipper(0, &m_pClipper, NULL) != DD_OK) {
ReleaseDrawSurface();
}
if (m_pClipper) {
if (m_pClipper->SetHWnd(0, GetPane()) != DD_OK) {
ReleaseDrawSurface();
}
else {
if (m_lpDDSPrimary->SetClipper(m_pClipper) != DD_OK) {
ReleaseDrawSurface();
}
}
}
if (m_lpDD->CreateClipper(0, &m_pOffScreenClipper, NULL) != DD_OK) {
ReleaseDrawSurface();
}
else {
if (m_lpDDSBack->SetClipper(m_pOffScreenClipper) != DD_OK) {
ReleaseDrawSurface();
}
}
}
}
}
#endif
}
#ifdef DDRAW
void CWinCX::SetClipOnDrawSurface(LPDIRECTDRAWSURFACE surface, HRGN hClipRgn)
{
if (m_offScreenDC) {
if (hClipRgn == FE_NULL_REGION) {
::SelectClipRgn(m_offScreenDC, NULL);
}
else {
::SelectClipRgn(m_offScreenDC, hClipRgn);
}
}
}
void CWinCX::RestoreAllDrawSurface()
{
#ifdef XP_WIN32
if( m_lpDDSPrimary->Restore() == DD_OK ) {
if (m_lpDDSBack->Restore() == DD_OK) {
return;
}
}
// if the surface can not be restore.
TRACE("There is something wrong with DirectDraw surface.\n");
ReleaseDrawSurface();
#endif
}
void CWinCX::ReleaseDrawSurface()
{
#ifdef XP_WIN32
if (m_offScreenDC) {
ReleaseOffscreenSurfDC();
m_offScreenDC = 0;
}
if (m_pOffScreenClipper) {
m_pOffScreenClipper->Release();
m_pOffScreenClipper = NULL;
}
if (m_pClipper) {
m_pClipper->Release();
m_pClipper = NULL;
}
if (m_lpDDSPrimary) {
m_lpDDSPrimary->Release();
m_lpDDSPrimary = NULL;
}
if (m_lpDDSBack) {
m_lpDDSBack->Release();
m_lpDDSBack = NULL;
}
if (mg_pPal) {
mg_pPal->Release();
mg_pPal = NULL;
}
if (m_lpDD) {
m_lpDD->Release();
m_lpDD = NULL;
}
HDC hdc = GetContextDC();
CDCCX::SetUseDibPalColors(!IsPrintContext() && (::GetDeviceCaps(hdc, RASTERCAPS) & RC_PALETTE));
ReleaseContextDC(hdc);
#endif // WIN32
}
#endif // DDRAW
void CWinCX::ClearFrame() {
m_pFrame = NULL;
// Go through all our immediate children, clearing
// their frame.
// They will in turn call thier children, if present.
MWContext *pChild;
XP_List *pTraverse = GetContext()->grid_children;
while (pChild = (MWContext*)XP_ListNextObject(pTraverse)) {
WINCX(pChild)->ClearFrame();
}
}
// If we're a grid cell, we need to return the parent's palette.
HPALETTE CWinCX::GetPalette() const {
if(IsGridCell() == TRUE && GetParentContext() != NULL) {
ASSERT(ABSTRACTCX(GetParentContext())->IsWindowContext());
return(WINCX(GetParentContext())->GetPalette());
}
// Return the normal stuff.
return(CDCCX::GetPalette());
}
void CWinCX::DestroyContext() {
if(IsDestroyed() == FALSE) {
ResetToolTipImg();
// Deactivate any embedded items.
OnDeactivateEmbedCX();
// Release our modality if any is in effect.
POSITION rTraverse = m_cplModalOver.GetHeadPosition();
while(rTraverse) {
HWND hwndOwner;
// We stored the HWND of the window we're being modal over
// Retrieve it.
hwndOwner = (HWND)m_cplModalOver.GetNext(rTraverse);
if (::IsWindow(hwndOwner)) {
// Must make sure no one else thinks they are modal over the
// window before we go off and enable it.
int iTraverse = 1;
BOOL bEnable = TRUE;
MWContext *pMW = NULL;
XP_List *pTraverse = XP_GetGlobalContextList();
while (pMW = (MWContext *)XP_ListNextObject(pTraverse)) {
if(ABSTRACTCX(pMW) && ABSTRACTCX(pMW)->IsWindowContext()) {
CWinCX *pCX = WINCX(pMW);
if(this == pCX) {
// Exclude ourselves from the check.
continue;
}
POSITION rInsane = pCX->m_cplModalOver.GetHeadPosition();
HWND hCheck = NULL;
while(rInsane) {
hCheck = (HWND)pCX->m_cplModalOver.GetNext(rInsane);
if(hCheck == hwndOwner) {
// Someone else will enable it later.
bEnable = FALSE;
break;
}
}
}
if(FALSE == bEnable) {
// No need to continue if decided already.
break;
}
}
if(bEnable) {
::EnableWindow(hwndOwner, TRUE);
}
}
}
m_cplModalOver.RemoveAll();
// Perform our close callbacks.
if(!m_cplCloseCallbacks.IsEmpty()) {
POSITION rFuncs = m_cplCloseCallbacks.GetHeadPosition();
POSITION rArgs = m_cplCloseCallbackArgs.GetHeadPosition();
void (*pCloseCallback)(void *) = NULL;
void *pCloseCallbackArg = NULL;
while(rFuncs) {
pCloseCallback = (void (*)(void *))m_cplCloseCallbacks.GetNext(rFuncs);
pCloseCallbackArg = m_cplCloseCallbackArgs.GetNext(rArgs);
if(pCloseCallback != NULL) {
pCloseCallback(pCloseCallbackArg);
}
}
m_cplCloseCallbacks.RemoveAll();
m_cplCloseCallbackArgs.RemoveAll();
}
MWContext *pContext = GetContext();
//JavaScript has lifetime-linked windows which must close now too
if (pContext->js_dependent_list) {
MWContext *pDepContext;
XP_List *pTraverse = pContext->js_dependent_list;
while (pDepContext = (MWContext *)XP_ListNextObject(pTraverse)) {
pDepContext->js_parent = 0;
FE_DestroyWindow(pDepContext);
}
XP_ListDestroy(pContext->js_dependent_list);
pContext->js_dependent_list = NULL;
}
if (pContext->js_parent) {
if (XP_ListCount(pContext->js_parent->js_dependent_list) == 1) {
XP_ListDestroy(pContext->js_parent->js_dependent_list);
pContext->js_parent->js_dependent_list=NULL;
}
else {
XP_ListRemoveObject(pContext->js_parent->js_dependent_list, pContext);
}
}
// Call the base. This will delete the object !!
// To make sure that m_pPal is not being selected to a dc
HDC hdc = GetContextDC();
::SelectPalette(hdc, (HPALETTE)::GetStockObject(DEFAULT_PALETTE), FALSE);
// Get rid of the home grown DC that we created.
if(GetPane() != NULL) {
#ifdef DDRAW
ReleaseDrawSurface();
#endif
ReleaseContextDC(hdc);
}
#ifdef JAVA
//
// Discard the events pending for the context...
//
LJ_DiscardEventsForContext(pContext);
#endif /* JAVA */
}
CPaneCX::DestroyContext();
}
void CWinCX::OnDeactivateEmbedCX() {
CGenericView *pView = GetView();
if(pView != NULL && m_pSelected != NULL) {
// Obtain the plugin structure.
NPEmbeddedApp *pPluginShim = (NPEmbeddedApp *)m_pSelected->FE_Data;
// Make sure it is not a plugin, but an OLE container item.
if(pPluginShim != NULL && wfe_IsTypePlugin(pPluginShim) == FALSE) {
// Get the container item, and deactivate it.
CNetscapeCntrItem *pItem = (CNetscapeCntrItem *)pPluginShim->fe_data;
if(pItem != NULL && pItem->IsInPlaceActive() == TRUE) {
TRY {
pItem->Deactivate();
}
CATCH(CException, e) {
// Something went wrong in OLE (other app down).
// No complicated handling here, just keep running.
}
END_CATCH
}
}
// Clear that nothing is currently selected.
m_pSelected = NULL;
}
}
#ifdef EDITOR
void CWinCX::Scroll(int iBars, UINT uSBCode, UINT uPos, HWND hCtrl, UINT uTimes)
{
int32 lOldOrgY = GetOriginY();
int32 lOldOrgX = GetOriginX();
CPaneCX::Scroll(iBars, uSBCode, uPos, hCtrl, uTimes);
if(lOldOrgY != GetOriginY() || lOldOrgX != GetOriginX()) {
if( EDT_IS_EDITOR(GetContext()) ){
EDT_WindowScrolled( GetContext() );
}
}
}
#endif
#ifdef DDRAW
void CWinCX::ScrollWindow(int x, int y)
{
RECT source, dest;
dest.top = m_physicWinRect.top;
dest.bottom = m_physicWinRect.bottom;
dest.left = m_physicWinRect.left;
dest.right = m_physicWinRect.right;
source.top = m_physicWinRect.top;
source.bottom = m_physicWinRect.bottom;
source.left = m_physicWinRect.left;
source.right = m_physicWinRect.right;
RECT updateRect;
updateRect.top = m_physicWinRect.top;
updateRect.bottom = m_physicWinRect.bottom;
updateRect.left = m_physicWinRect.left;
updateRect.right = m_physicWinRect.right;
if (y == 0) { // scrolling horz
if (x > 0) {// scroll right.
dest.left += x;
source.right -= x;
}
else {
dest.right += x;
source.left -= x;
}
if (m_physicWinRect.left == dest.left) {
updateRect.left = dest.right;
}
else
updateRect.left = m_physicWinRect.left;
updateRect.right = updateRect.left + abs(x);
}
else if (x == 0) { // scroll vert.
if (y > 0) {// scroll down.
dest.top += y;
source.bottom -= y;
}
else {
dest.bottom += y;
source.top -= y;
}
if (m_physicWinRect.top == dest.top) {
updateRect.top = dest.bottom;
}
else
updateRect.top = m_physicWinRect.top;
updateRect.bottom = updateRect.top + abs(y);
}
if (y < 0 || x < 0) {
if(IsVScrollBarOn() ) {
updateRect.left -= sysInfo.m_iScrollWidth;
updateRect.right -= sysInfo.m_iScrollWidth;
}
if(IsHScrollBarOn()) {
updateRect.top -= sysInfo.m_iScrollWidth;
updateRect.bottom -= sysInfo.m_iScrollWidth;
}
}
::OffsetRect(&updateRect, -GetWindowsXPos(), -GetWindowsYPos());
HRESULT err = m_lpDDSBack->ReleaseDC(m_offScreenDC);
::OffsetRect(&dest, -GetWindowsXPos(), -GetWindowsYPos());
err = m_lpDDSBack->Blt(&dest, m_lpDDSPrimary, &source, DDBLT_WAIT, NULL);
if (err == DDERR_SURFACELOST) {
RestoreAllDrawSurface();
err = m_lpDDSBack->Blt(&dest, m_lpDDSPrimary, &source, DDBLT_WAIT, NULL);
}
m_lpDDSBack->GetDC(&m_offScreenDC);
m_ScrollWindow = TRUE;
#ifdef DEBUG_mhwang
TRACE("Scroll Window\n");
#endif
m_pAlternateDC = m_offScreenDC;
RefreshArea(m_offScreenDC, updateRect.left + m_lOrgX, updateRect.top + m_lOrgY,
updateRect.right - updateRect.left,
updateRect.bottom - updateRect.top);
RECT prcUpdate;
// ::ScrollWindowEx(GetPane(), x, (int) y, NULL, NULL, NULL, &prcUpdate, SW_SCROLLCHILDREN);
if (m_lpDDSPrimary ) {
LTRB rect(m_physicWinRect);
rect.left -= GetWindowsXPos();
rect.right -= GetWindowsXPos();
rect.top -= GetWindowsYPos();
rect.bottom -= GetWindowsYPos();
FE_Region hCurClip = GetDrawingClip();
HDC hdc;
m_lpDDSPrimary->GetDC(&hdc);
::SelectClipRgn(hdc, NULL);
m_lpDDSPrimary->ReleaseDC(hdc);
BltToScreen(rect, NULL);
}
m_pAlternateDC = 0;
m_ScrollWindow = FALSE;
}
#endif
// This function get's called when the window moves around.
void CWinCX::OnMoveCX() {
// WARNING:m_crWindowRect will be invalid until next AftWMSize!
// Go through all our immediate children, telling them their screen location
// has changed.
// They will in turn call thier children, if present.
MWContext *pChild;
XP_List *pTraverse = GetContext()->grid_children;
while (pChild = (MWContext*)XP_ListNextObject(pTraverse)) {
WINCX(pChild)->OnMoveCX();
}
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOVE;
CFrameWnd *pFWnd = GetFrame()->GetFrameWnd();
if(pFWnd) {
CRect crFrame;
pFWnd->GetWindowRect(crFrame);
event->x = (int32)crFrame.left;
event->y = (int32)crFrame.top;
event->screenx = (int32)crFrame.left;
event->screeny = (int32)crFrame.top;
ET_SendEvent(GetContext(), 0, event,
0, 0);
}
#ifdef DDRAW
CalcWinPos();
#endif
}
static void
wfe_ResizeFullPagePlugin(MWContext* pContext, int32 lWidth, int32 lHeight)
{
ASSERT(pContext);
NPWindow* npWindow = pContext->pluginList->wdata;
ASSERT(npWindow);
if (npWindow) {
npWindow->width = lWidth;
npWindow->height = lHeight;
::SetWindowPos((HWND)npWindow->window,
NULL,
0,
0,
(int)npWindow->width,
(int)npWindow->height,
SWP_NOZORDER | SWP_NOMOVE | SWP_NOACTIVATE);
NPL_EmbedSize(pContext->pluginList);
}
}
#ifdef DDRAW
void CWinCX::CreateAndLockOffscreenSurface(RECT& rect)
{
// create new offscreen surface.
if (m_lpDDSPrimary) {
if (m_lpDDSBack) {
m_lpDDSBack->ReleaseDC(m_offScreenDC);
m_offScreenDC = 0;
m_lpDDSBack->Release();
}
m_lpDDSBack = CreateOffscreenSurface(rect);
RestoreAllDrawSurface();
if (m_lpDDSBack)
m_lpDDSBack->GetDC(&m_offScreenDC);
}
}
LPDIRECTDRAWSURFACE CWinCX::CreateOffscreenSurface(RECT& rect)
{
LPDIRECTDRAWSURFACE retval;
// create new offscreen surface.
ZeroMemory((void*)&ddsd, sizeof(DDSURFACEDESC));
ddsd.dwSize = sizeof( ddsd );
// Create a offscreen bitmap.
ddsd.dwFlags = DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH;
ddsd.ddsCaps.dwCaps = DDSCAPS_OFFSCREENPLAIN;
ddsd.dwHeight = rect.bottom - rect.top;
ddsd.dwWidth = rect.right - rect.left;
if (m_lpDD->CreateSurface( &ddsd, &retval, NULL )!=DD_OK) {
ReleaseDrawSurface();
return NULL;
}
else
return retval;
}
#endif
void CWinCX::AftWMSize(PaneMessage *pMessage)
{
// Purpose: Informs the context that the size of the displayable area has changed.
// Arguments: As OnSize in MFC
// Returns: void
// Comments: Pretty much handles every want of a resizeable window.
// Revision History:
// 07-13-95 created GABby
//
// Call the base.
// Preserve old size for logic in this function.
int32 lOldWidth = GetWidth();
int32 lOldHeight = GetHeight();
CPaneCX::AftWMSize(pMessage);
UINT uType = (UINT)pMessage->wParam;
int iX = LOWORD(pMessage->lParam);
int iY = HIWORD(pMessage->lParam);
// Wether or not we should call NiceReload before we return from this function.
// We stall the call, as we need to update all of our state size variables
// before we start a load which may depend upon these state variables
// to be correct for display purposes and for turning on and off scroll
// bars etc.
BOOL bNiceReload = FALSE;
// Block flailing random message we receive when a new browser
// window is opened and we're minimized.
if(GetFrame() && GetFrame()->GetFrameWnd() &&
GetFrame()->GetFrameWnd()->IsIconic() &&
uType == 0 && iX == 0 && iY == 0) {
// Stop the stupid windows behavior.
return;
}
// There is no window.
if(!GetPane() )
return;
// Update our idea of where we exist.
CRect crNewRect(0, 0, 0, 0);
::GetWindowRect(GetPane(), crNewRect);
#ifdef DDRAW
// create new offscreen surface.
if (m_lpDDSPrimary) {
CreateAndLockOffscreenSurface(crNewRect);
}
// mwh - reset our windows's screen position. so it will be calcuate
// in DisplayPixmap again.
m_physicWinRect.Empty();
#endif
if(IsGridCell() && !m_crWindowRect.EqualRect(crNewRect)) {
// NCAPI sizing support.
CWindowChangeItem::Sizing(GetContextID(), crNewRect.left, crNewRect.top, crNewRect.Width(), crNewRect.Height());
}
// Don't do this on anything but restoration and maximization.
// We don't care if we're minimized.
if(uType != SIZE_MAXIMIZED && uType != SIZE_RESTORED) {
TRACE("Don't handle size messages of type %u\n", uType);
return;
}
// Update what our document thinks if In Place.
if(GetDocument() && GetDocument()->IsInPlaceActive() && GetPane()) {
int32 lIPWidth = Twips2MetricX(iX);
int32 lIPHeight = Twips2MetricY(iY);
// quasi-hack alert.
// When the scroll bars are on, our twips to metrics conversions
// are off by 1 pixel in the direction of the scroller.
// Adjust.
// This seems reverse logic, but is correct (horiz scroll bar
// shortens height of window when present).
if(IsHScrollBarOn()) {
lIPHeight -= Twips2MetricX(1);
}
if(IsVScrollBarOn()) {
lIPWidth -= Twips2MetricY(1);
}
GetDocument()->m_csViewExtent = CSize(CASTINT(lIPWidth), CASTINT(lIPHeight));
TRACE("InPlace Document extents now %ld,%ld\n", lIPWidth, lIPHeight);
}
CGenericView *pView = GetView();
ASSERT(pView);
// We can never allow resize when there are frame cells and we are in
// print preview. The print preview state holds a pointer to
// one of the views in the frameset. Once print preview is
// completed, the code will attempt to set that view to be the
// active view. If deleted (as from a reload/resize), then we crash.
// Also, if only the Y value changed, then just adjust the iHeight.
// This doesn't apply to grid parents OR Composer, as they need to reload in
// all resize cases.
if( !EDT_IS_EDITOR(GetContext()) &&
((IsGridParent() && pView->IsInPrintPreview()) ||
(GetWidth() == lOldWidth && IsGridParent() == FALSE))
) {
if(ContainsFullPagePlugin()) {
wfe_ResizeFullPagePlugin(GetContext(), GetWidth(), GetHeight());
}
{
MWContext *context = GetContext();
/* Mark the entire window dirty and force the compositor to composite now for
the case when the user resizes the top or bottom edge of the window.
This can be a potential performance problem!. Be careful.
*/
if (context->compositor && GetHeight() != lOldHeight)
{
CL_Compositor *compositor = context->compositor;
XP_Rect rect;
CL_OffscreenMode save_offscreen_mode;
rect.left = 0;
rect.top = 0;
rect.right = GetWidth();
rect.bottom = GetHeight();
CL_UpdateDocumentRect(compositor,
&rect, (PRBool)FALSE);
/* Temporarily force drawing to use the offscreen buffering area to reduce
flicker when resizing. (If no offscreen store is allocated, this code will
have no effect, but it will do no harm.) */
save_offscreen_mode = CL_GetCompositorOffscreenDrawing(compositor);
CL_SetCompositorOffscreenDrawing(compositor, CL_OFFSCREEN_ENABLED);
CL_CompositeNow(compositor);
CL_SetCompositorOffscreenDrawing(compositor, save_offscreen_mode);
/* Call Win32 API call to validate entire windows so that the WM_PAINT message
that follows a resize does not force a repaint of the window. The repaint
happens via the compositor in the CL_UpdateDocumentRect call above */
::ValidateRect( GetPane(), NULL );
}
}
return;
}
// Check the history to see if what we have is a binary image,
// or if this is a view-source URL.
// Don't reload if this is the case.
History_entry *pEntry = SHIST_GetCurrent(&(GetContext()->hist));
if(pEntry != NULL &&
(pEntry->is_binary ||
(pEntry->address && 0 == strncmp(pEntry->address, "view-source:", 12)))) {
if(ContainsFullPagePlugin()) {
// There can be only one plugin if it is full page
wfe_ResizeFullPagePlugin(GetContext(), GetWidth(), GetHeight());
}
return;
}
// Have any embedded items update.
CNetscapeCntrItem *pItem = (CNetscapeCntrItem *)GetDocument()->GetInPlaceActiveItem(pView);
if(pItem != NULL) {
pItem->SetItemRects();
}
#ifdef EDITOR
if( EDT_IS_EDITOR(GetContext()) ){
EDT_RefreshLayout( GetContext() );
}
else
#endif // EDITOR
{
// If this is edit_view_source, then we can't reload because the
// data didn't come from an acutal URL
if( !GetContext()->edit_view_source_hack ){
// Reload, we need to relayout everything to the new dimenstions.
if(m_crWindowRect.Width() != crNewRect.Width() ||
m_crWindowRect.Height() != crNewRect.Height()) {
bNiceReload = TRUE;
}
}
}
m_crWindowRect = crNewRect;
// Now that all variables are updated and in sync, we can call
// NiceReload if needed.
if(bNiceReload && XP_DOCID(GetDocumentContext())) {
/* MWContext *context = GetContext(); */
// Instead of re-loading from scratch, we want to pass new width, height of window
// to the re-layout routine.
LO_RelayoutOnResize(GetDocumentContext(), GetWidth(), GetHeight(), m_lLeftMargin, m_lTopMargin);
/* Call Win32 API call to validate entire window so that the WM_PAINT message
that follows a resize does not force a repaint of the window. The repaint
happens in the LO_RelayoutOnResize call above */
::ValidateRect( GetPane(), NULL );
/*
NiceResizeReload();
*/
}
}
#ifdef EDITOR
void CWinCX::PreWMErasebkgnd(PaneMessage *pMessage)
{
MWContext *pMWContext = GetContext();
if( pMWContext && EDT_IS_EDITOR(pMWContext) ){
WFE_HideEditCaret(pMWContext);
}
CPaneCX::PreWMErasebkgnd(pMessage);
if( pMWContext && EDT_IS_EDITOR(pMWContext) ){
WFE_ShowEditCaret(pMWContext);
}
}
#endif
BOOL CWinCX::EraseTextBkgnd(HDC pDC, RECT& cRect, LO_TextStruct* pText)
{
// If there's a background color specified then use it
return CDCCX::_EraseBkgnd(pDC, cRect, GetOriginX(), GetOriginY(),
pText->text_attr->no_background ? NULL : &pText->text_attr->bg);
}
#ifdef LAYERS
BOOL CWinCX::HandleLayerEvent(CL_Layer * pLayer, CL_Event * pEvent)
{
XY point(pEvent->x, pEvent->y); // Event location, in layer coordinates
BOOL bReturn = TRUE;
fe_EventStruct *pFEEvent = (fe_EventStruct *)pEvent->fe_event;
UINT uFlags, nFlags, nRepCnt, nChar;
CPoint cpPoint; // Event location, in screen coordinates
LO_Element *pElement;
CFormElement * pFormElement;
if (CL_IS_MOUSE_EVENT(pEvent)) {
if (pFEEvent) {
uFlags = pFEEvent->uFlags;
cpPoint.x = pFEEvent->x;
cpPoint.y = pFEEvent->y;
}
else {
// This is a synthesized event and we need to fill in
// the FE part. We can just used the uFlags from the
// previous mouse event. Should we know all the information
// to create the cpPoint?
uFlags = m_uMouseFlags;
int32 layer_x_offset = CL_GetLayerXOrigin(pLayer);
int32 layer_y_offset = CL_GetLayerYOrigin(pLayer);
cpPoint.x = CASTINT(pEvent->x + layer_x_offset - m_lOrgX);
cpPoint.y = CASTINT(pEvent->y + layer_y_offset - m_lOrgY);
}
}
if (CL_IS_KEY_EVENT(pEvent)) {
if (pFEEvent) {
nFlags = HIWORD(pFEEvent->fe_modifiers);
nRepCnt = LOWORD(pFEEvent->fe_modifiers);
nChar = pFEEvent->nChar;
}
}
switch(pEvent->type) {
case CL_EVENT_MOUSE_BUTTON_DOWN:
if (pEvent->which == 1)
OnLButtonDownForLayerCX(uFlags, cpPoint, point, pLayer);
// The right button is passed to the view for popup stuff
else if (pEvent->which == 3) {
if(!OnRButtonDownForLayerCX(uFlags, cpPoint,
point, pLayer)) {
CGenericView *pView = GetView();
if(pView) {
bReturn = pView->OnRButtonDownForLayer(uFlags, cpPoint,
point.x, point.y,
pLayer);
}
}
}
break;
case CL_EVENT_MOUSE_BUTTON_UP:
if (pEvent->which == 1)
OnLButtonUpForLayerCX(uFlags, cpPoint, point, pLayer, pFEEvent->pbReturnImmediately);
else if (pEvent->which == 3)
OnRButtonUpForLayerCX(uFlags, cpPoint, point, pLayer);
break;
case CL_EVENT_MOUSE_MOVE:
OnMouseMoveForLayerCX(uFlags, cpPoint, point, pLayer, pFEEvent->pbReturnImmediately);
break;
case CL_EVENT_MOUSE_BUTTON_MULTI_CLICK:
if (pEvent->which == 1)
OnLButtonDblClkForLayerCX(uFlags, cpPoint, point, pLayer);
else if (pEvent->which == 3)
OnRButtonDblClkForLayerCX(uFlags, cpPoint, point, pLayer);
break;
case CL_EVENT_KEY_FOCUS_GAINED:
bReturn = TRUE;
break;
case CL_EVENT_KEY_UP:
pElement = GetLayoutElement(point, pLayer);
// Check for form element and send event back to form element's class.
if (pElement != NULL && pElement->type == LO_FORM_ELE && pFEEvent->x == 1) {
switch (pElement->lo_form.element_data->type) {
case FORM_TYPE_TEXT:
case FORM_TYPE_PASSWORD:
case FORM_TYPE_TEXTAREA:
case FORM_TYPE_FILE:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
((CNetscapeEdit *)CEdit::FromHandlePermanent(pFormElement->GetRaw()))->OnEditKeyEvent(pEvent->type, pEvent->which, nRepCnt, nFlags);
break;
default:
break;
}
}
break;
case CL_EVENT_KEY_DOWN:
pElement = GetLayoutElement(point, pLayer);
// Check for form element and send event back to form element's class.
if (pElement != NULL && pElement->type == LO_FORM_ELE && pFEEvent->x == 1) {
switch (pElement->lo_form.element_data->type) {
case FORM_TYPE_TEXT:
case FORM_TYPE_PASSWORD:
case FORM_TYPE_TEXTAREA:
case FORM_TYPE_FILE:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
((CNetscapeEdit *)CEdit::FromHandlePermanent(pFormElement->GetRaw()))->OnEditKeyEvent(pEvent->type, pEvent->which, nRepCnt, nFlags);
break;
default:
break;
}
}
else {
switch(nChar) {
case ' ':
case VK_NEXT:
// page down
Scroll(SB_VERT, SB_PAGEDOWN, 0, NULL);
break;
case VK_BACK:
case VK_PRIOR:
// page up
Scroll(SB_VERT, SB_PAGEUP, 0, NULL);
break;
case VK_UP:
// line up
Scroll(SB_VERT, SB_LINEUP, 0, NULL);
break;
case VK_DOWN:
// line down
Scroll(SB_VERT, SB_LINEDOWN, 0, NULL);
break;
case VK_RIGHT:
// line right
Scroll(SB_HORZ, SB_LINERIGHT, 0, NULL);
break;
case VK_LEFT:
// line left
Scroll(SB_HORZ, SB_LINELEFT, 0, NULL);
break;
case VK_HOME:
if (::GetKeyState(VK_CONTROL) < 0)
Scroll(SB_VERT, SB_TOP, 0, NULL);
else
Scroll(SB_HORZ, SB_TOP, 0, NULL);
break;
case VK_END:
if (::GetKeyState(VK_CONTROL) < 0)
Scroll(SB_VERT, SB_BOTTOM, 0, NULL);
else
Scroll(SB_HORZ, SB_BOTTOM, 0, NULL);
break;
case VK_ESCAPE:
// escape, kill off any selected items.
if(m_pSelected != NULL) {
OnDeactivateEmbedCX();
}
break;
default:
break;
}
}
break;
case CL_EVENT_KEY_FOCUS_LOST:
default:
bReturn = FALSE;
break;
}
return bReturn;
}
BOOL CWinCX::HandleEmbedEvent(LO_EmbedStruct *embed, CL_Event *pEvent)
{
NPEvent npEvent;
fe_EventStruct *pFEEvent = (fe_EventStruct *)pEvent->fe_event;
NPEmbeddedApp *pEmbeddedApp = (NPEmbeddedApp *)embed->FE_Data;
if (CL_IS_MOUSE_EVENT(pEvent)) {
if (pFEEvent)
npEvent.wParam = pFEEvent->uFlags;
else
npEvent.wParam = m_uMouseFlags;
npEvent.lParam = MAKELONG(pEvent->x, pEvent->y);
}
else if (CL_IS_KEY_EVENT(pEvent)) {
npEvent.wParam = (uint32)pEvent->which;
npEvent.lParam = (uint32)pEvent->modifiers;
}
else if (pEvent->type == CL_EVENT_MOUSE_ENTER) {
npEvent.wParam = 0;
npEvent.lParam = MAKELONG(HTCLIENT, 0);
}
else {
npEvent.wParam = 0;
npEvent.lParam = 0;
}
switch(pEvent->type) {
case CL_EVENT_MOUSE_BUTTON_DOWN:
if (pEvent->which == 1) {
npEvent.event = WM_LBUTTONDOWN;
}
else if (pEvent->which == 2)
npEvent.event = WM_MBUTTONDOWN;
else
npEvent.event = WM_RBUTTONDOWN;
break;
case CL_EVENT_MOUSE_BUTTON_UP:
if (pEvent->which == 1)
npEvent.event = WM_LBUTTONUP;
else if (pEvent->which == 2)
npEvent.event = WM_MBUTTONUP;
else
npEvent.event = WM_RBUTTONUP;
break;
case CL_EVENT_MOUSE_MOVE:
npEvent.event = WM_MOUSEMOVE;
break;
case CL_EVENT_MOUSE_BUTTON_MULTI_CLICK:
if (pEvent->which == 1)
npEvent.event = WM_LBUTTONDBLCLK;
else if (pEvent->which == 2)
npEvent.event = WM_MBUTTONDBLCLK;
else
npEvent.event = WM_RBUTTONDBLCLK;
break;
case CL_EVENT_KEY_UP:
npEvent.event = WM_KEYUP;
break;
case CL_EVENT_KEY_DOWN:
npEvent.event = WM_KEYDOWN;
break;
case CL_EVENT_MOUSE_ENTER:
npEvent.event = WM_SETCURSOR;
break;
case CL_EVENT_KEY_FOCUS_GAINED:
npEvent.event = WM_SETFOCUS;
break;
case CL_EVENT_KEY_FOCUS_LOST:
npEvent.event = WM_KILLFOCUS;
break;
default:
return FALSE;
}
return (BOOL)NPL_HandleEvent(pEmbeddedApp, &npEvent, (void*)npEvent.wParam);
}
#endif /* LAYERS */
BOOL CWinCX::OnKeyUp(UINT nChar, UINT nRepCnt, UINT nFlags)
{
BOOL bReturn = FALSE;
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return bReturn;
}
#ifdef LAYERS
/*
* If there's a compositor and someone has keyboard focus.
* Note that if noone has event focus, we set the event focus
* to the main document.
*/
if (GetContext()->compositor) {
if (CL_IsKeyEventGrabber(GetContext()->compositor, NULL) &&
CL_GetCompositorRoot(GetContext()->compositor)) {
CL_GrabKeyEvents(GetContext()->compositor, CL_GetLayerChildByName(
CL_GetCompositorRoot(GetContext()->compositor), LO_BODY_LAYER_NAME));
}
if (!CL_IsKeyEventGrabber(GetContext()->compositor, NULL)) {
CL_Event event;
fe_EventStruct fe_event;
// Convert the point to something we understand.
XY Point;
WORD asciiChar = 0;
ResolvePoint(Point, m_cpMMove);
if (!EDT_IS_EDITOR(GetContext())) {
BYTE kbstate[256];
GetKeyboardState(kbstate);
#ifdef WIN32
ToAscii(nChar, nFlags & 0xff, kbstate, &asciiChar, 0);
#else
ToAscii(nChar, nFlags & 0xff, kbstate, (DWORD*)&asciiChar, 0);
#endif
}
fe_event.fe_modifiers = MAKELONG(nRepCnt, nFlags);
fe_event.nChar = nChar;
fe_event.x = 0;
event.type = CL_EVENT_KEY_UP;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.which = asciiChar;
event.modifiers = (GetKeyState(VK_SHIFT) < 0 ? EVENT_SHIFT_MASK : 0)
| (GetKeyState(VK_CONTROL) < 0 ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.x = Point.x;
event.y = Point.y;
bReturn = (BOOL)CL_DispatchEvent(GetContext()->compositor, &event);
}
}
#endif /* LAYERS */
return bReturn;
}
BOOL CWinCX::OnKeyDown(UINT nChar, UINT nRepCnt, UINT nFlags)
{
BOOL bReturn = FALSE;
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return bReturn;
}
#ifdef LAYERS
/*
* If there's a compositor and someone has keyboard focus.
* Note that if noone has event focus, we set the event focus
* to the main document.
*/
if (GetContext()->compositor) {
if (CL_IsKeyEventGrabber(GetContext()->compositor, NULL) &&
CL_GetCompositorRoot(GetContext()->compositor)) {
CL_GrabKeyEvents(GetContext()->compositor, CL_GetLayerChildByName(
CL_GetCompositorRoot(GetContext()->compositor), LO_BODY_LAYER_NAME));
}
if (!CL_IsKeyEventGrabber(GetContext()->compositor, NULL)) {
CL_Event event;
fe_EventStruct fe_event;
// Convert the point to something we understand.
XY Point;
WORD asciiChar = 0;
ResolvePoint(Point, m_cpMMove);
if (!EDT_IS_EDITOR(GetContext())) {
BYTE kbstate[256];
GetKeyboardState(kbstate);
#ifdef WIN32
ToAscii(nChar, nFlags & 0xff, kbstate, &asciiChar, 0);
#else
ToAscii(nChar, nFlags & 0xff, kbstate, (DWORD*)&asciiChar, 0);
#endif
}
fe_event.fe_modifiers = MAKELONG(nRepCnt, nFlags);
fe_event.nChar = nChar;
fe_event.x = 0;
event.type = CL_EVENT_KEY_DOWN;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.which = asciiChar;
event.modifiers = (GetKeyState(VK_SHIFT) < 0 ? EVENT_SHIFT_MASK : 0)
| (GetKeyState(VK_CONTROL) < 0 ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.x = Point.x;
event.y = Point.y;
event.data = nFlags>>14 & 1;//Bit represeting key repetition
bReturn = (BOOL)CL_DispatchEvent(GetContext()->compositor, &event);
}
}
#endif /* LAYERS */
return bReturn;
}
void CWinCX::OnLButtonDblClkCX(UINT uFlags, CPoint cpPoint) {
// Only do this if clicking is enabled.
if(IsClickingEnabled() == FALSE) {
return;
}
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return;
}
// Remember....
m_LastMouseEvent = m_LBDClick;
m_cpLBDClick = cpPoint;
m_uMouseFlags = uFlags;
// Convert the point to something we understand.
XY Point;
ResolvePoint(Point, cpPoint);
#ifdef LAYERS
if (GetContext()->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
event.type = CL_EVENT_MOUSE_BUTTON_MULTI_CLICK;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.x = Point.x;
event.y = Point.y;
event.modifiers = (uFlags & MK_SHIFT ? EVENT_SHIFT_MASK : 0)
| (uFlags & MK_CONTROL ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.which = 1;
event.data = 2;
CL_DispatchEvent(GetContext()->compositor,
&event);
}
else
OnLButtonDblClkForLayerCX(uFlags, cpPoint, Point, NULL);
// Have the mouse timer handler do some dirty work.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
return;
}
void
CWinCX::OnLButtonDblClkForLayerCX(UINT uFlags, CPoint& cpPoint,
XY& Point, CL_Layer *layer)
{
// With LAYERS turned on, the orginal method
// OnLButtonDblClkCX is separated into two methods,
// one of which is a per-layer method.
#endif /* LAYERS */
// Process any embed activation.
#ifdef LAYERS
LO_Element *pElement = GetLayoutElement(Point, layer);
#else
LO_Element *pElement = GetLayoutElement(Point);
#endif
if (pElement != NULL && pElement->type == LO_FORM_ELE &&
(Point.x - pElement->lo_form.x - pElement->lo_form.x_offset < pElement->lo_form.width) &&
(Point.x - pElement->lo_form.x - pElement->lo_form.x_offset > 0) &&
(Point.y - pElement->lo_form.y - pElement->lo_form.y_offset < pElement->lo_form.height) &&
(Point.y - pElement->lo_form.y - pElement->lo_form.y_offset > 0)) {
CFormElement * pFormElement;
CNetscapeButton *pButton;
switch (pElement->lo_form.element_data->type) {
case FORM_TYPE_BUTTON:
case FORM_TYPE_RESET:
case FORM_TYPE_SUBMIT:
case FORM_TYPE_CHECKBOX:
case FORM_TYPE_RADIO:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
((CNetscapeButton *)CButton::FromHandlePermanent(pFormElement->GetRaw()))
->OnButtonEvent(CL_EVENT_MOUSE_BUTTON_MULTI_CLICK, uFlags, cpPoint);
break;
case FORM_TYPE_FILE:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
pButton = (CNetscapeButton *)pFormElement->GetSecondaryWidget();
if (pButton)
pButton->OnButtonEvent(CL_EVENT_MOUSE_BUTTON_MULTI_CLICK, uFlags, cpPoint);
break;
default:
break;
}
return;
}
if(pElement != NULL && pElement->type == LO_EMBED) {
// Any previous activated embed was deactivated in the button down event.
// However, we should not reactivate an already active item.
NPEmbeddedApp* pEmbeddedApp = (NPEmbeddedApp*)pElement->lo_embed.FE_Data;
ASSERT(pEmbeddedApp);
CNetscapeCntrItem *pItem = (CNetscapeCntrItem *)pEmbeddedApp->fe_data;
if(pItem != NULL) {
if(pItem->m_bBroken == FALSE && pItem->m_bDelayed == FALSE &&
pItem->m_bLoading == FALSE) {
if(pItem->IsInPlaceActive() == FALSE) {
// Set the active item.
// This value is held in the view for now....
CGenericView *pView = GetView();
if(pView) {
m_pSelected = &(pElement->lo_embed);
long lVerb = OLEIVERB_PRIMARY;
if(uFlags & MK_CONTROL) {
lVerb = OLEIVERB_OPEN;
}
pView->BeginWaitCursor();
TRY {
pItem->Activate(lVerb, pView);
}
CATCH(CException, e) {
// Object wouldn't activate or something went wrong,
// and almost caused us to go down with it.
m_pSelected = NULL;
}
END_CATCH
pView->EndWaitCursor();
// If it's not in place active at this point, there's no need
// to keep track of it.
if(m_pSelected != NULL && pItem->IsInPlaceActive() == FALSE) {
m_pSelected = NULL;
}
}
}
}
}
// Dbl-click is not the same as holding mouse down
m_bLBDown = FALSE;
m_bLBUp = TRUE;
}
else {
if(GetPane()) {
::SetCapture(GetPane());
}
#ifdef EDITOR
if ( EDT_IS_EDITOR(GetContext()) )
{
// Let the editor handle the double-click
EDT_DoubleClick(GetContext(), Point.x, Point.y);
// Dbl-click is NOT the same as holding mouse down
m_bLBDown = FALSE;
m_bLBUp = TRUE;
}
else
{
#ifdef LAYERS
LO_DoubleClick(GetDocumentContext(), Point.x, Point.y, layer);
#else
LO_DoubleClick(GetDocumentContext(), Point.x, Point.y);
#endif /* LAYERS */
// Double-click is the same as holding mouse down when
// we're selecting.
//cmanske: WHY??? DOES THE BROWSER NEED THIS? BAD FOR EDITOR!
#endif // EDITOR
m_bLBDown = TRUE;
m_bLBUp = FALSE;
#ifdef EDITOR
}
#endif
}
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
}
BOOL CWinCX::PtInSelectedCell(CPoint &DocPoint, LO_CellStruct *cell,
BOOL &bContinue, LO_Element *start_element,
LO_Element *end_element)
{
BOOL bPtInRegion = FALSE;
for ( LO_Any_struct * element = (LO_Any_struct *)cell->cell_list; ;
element = (LO_Any_struct *)(element->next) ) {
if( element == 0 ){
bContinue = TRUE;
return FALSE;
}
// Linefeed rect is from end of text to right ledge,
// so lets ignore it
if ( element->type != LO_LINEFEED ) {
if ( element->type == LO_TEXT &&
(element == (LO_Any_struct*)start_element ||
element == (LO_Any_struct*)end_element) ) {
// With 1st and last text elements, we need to
// account for character offsets from start or end of selection
LO_TextStruct *text = (LO_TextStruct*)element;
LTRB rect;
// We may have a null text element in Tables,
// so use closest non-null text element
if( text->text == NULL){
if( text->prev != NULL && text->prev->type == LO_TEXT ){
text = (LO_TextStruct*)text->prev;
} else if( text->next != NULL && text->next->type == LO_TEXT ){
text = (LO_TextStruct*)text->next;
}
}
if( text->text ){
ResolveElement( rect, text,
(int)(text->x + text->x_offset), // Start location
(int32)(text->sel_start),
(int32)(text->sel_end), FALSE );
int x = CASTINT(DocPoint.x - GetOriginX());
int y = CASTINT(DocPoint.y - GetOriginY());
bPtInRegion = x > rect.left &&
x < rect.right &&
y > rect.top &&
y < rect.bottom;
} else {
bPtInRegion = FALSE;
}
}
else if (element->type == LO_CELL) {
bPtInRegion = PtInSelectedCell(DocPoint,
(LO_CellStruct *)element,
bContinue, start_element,
end_element);
}
else if (element->type != LO_TABLE) {
// Get the rect surrounding selected element,
CRect cRect;
cRect.left = CASTINT(element->x + element->x_offset);
cRect.top = CASTINT(element->y + element->y_offset);
cRect.right = CASTINT(cRect.left + element->width);
cRect.bottom = CASTINT(cRect.top + element->height);
bPtInRegion = cRect.PtInRect( DocPoint );
}
}
// We're done if we are in a rect or finished with last element
if ( bPtInRegion || !bContinue ||
element == (LO_Any_struct*)end_element ) {
break;
}
}
bContinue = FALSE;
return bPtInRegion;
}
// Test if point, such mouse cursor, is within the selected region
BOOL CWinCX::PtInSelectedRegion(CPoint cPoint, BOOL bConvertToDocCoordinates,
CL_Layer *layer)
{
BOOL bPtInRegion = FALSE;
BOOL bContinue = TRUE;
CPoint DocPoint;
if( bConvertToDocCoordinates ){
XY Point;
ResolvePoint(Point, cPoint);
DocPoint.x = CASTINT(Point.x);
DocPoint.y = CASTINT(Point.y);
} else {
DocPoint = cPoint;
}
int32 start_selection, end_selection;
LO_Element * start_element = NULL;
LO_Element * end_element = NULL;
CL_Layer *sel_layer = NULL;
int32 x_origin, y_origin, old_x_origin, old_y_origin;
// Start the search from the current selection location
LO_GetSelectionEndpoints(GetDocumentContext(),
&start_element,
&end_element,
&start_selection,
&end_selection,
&sel_layer);
if ( start_element == NULL ) {
return FALSE;
}
/*
* If the selection layer is different from the one in which
* the event occured, there isn't a match.
*/
if ( layer && (layer != sel_layer) ) {
return FALSE;
}
/*
* Temporarily change the drawing origin to that of the selection
* layer so that resolving the positions of the elements will
* result in correct coordinate space translation.
*/
if ( sel_layer ) {
x_origin = CL_GetLayerXOrigin(sel_layer);
y_origin = CL_GetLayerYOrigin(sel_layer);
}
else {
x_origin = 0;
y_origin = 0;
}
CDrawable *pDrawable = GetDrawable();
if (pDrawable) {
pDrawable->GetOrigin(&old_x_origin, &old_y_origin);
pDrawable->SetOrigin(x_origin, y_origin);
}
for ( LO_Any_struct * element = (LO_Any_struct *)start_element; ;
element = (LO_Any_struct *)(element->next) ) {
// KLUDGE: This prevents crashing when multiple selection
// within cells of a table
if( element == 0 ){
if ( pDrawable ) {
pDrawable->SetOrigin(old_x_origin, old_y_origin);
}
return FALSE;
}
// Linefeed rect is from end of text to right ledge,
// so lets ignore it
if ( element->type != LO_LINEFEED ) {
if ( element->type == LO_TEXT &&
(element == (LO_Any_struct*)start_element ||
element == (LO_Any_struct*)end_element) ) {
// With 1st and last text elements, we need to
// account for character offsets from start or end of selection
LO_TextStruct *text = (LO_TextStruct*)element;
LTRB rect;
// We may have a null text element in Tables,
// so use closest non-null text element
if( text->text == NULL){
if( text->prev != NULL && text->prev->type == LO_TEXT ){
text = (LO_TextStruct*)text->prev;
} else if( text->next != NULL && text->next->type == LO_TEXT ){
text = (LO_TextStruct*)text->next;
}
}
if( text->text ){
ResolveElement( rect, text,
(int)(text->x + text->x_offset), // Start location
(int32)(text->sel_start),
(int32)(text->sel_end), FALSE );
int x = CASTINT(DocPoint.x - GetOriginX());
int y = CASTINT(DocPoint.y - GetOriginY());
bPtInRegion = x > rect.left &&
x < rect.right &&
y > rect.top &&
y < rect.bottom;
} else {
bPtInRegion = FALSE;
}
}
else if (element->type == LO_CELL) {
bPtInRegion = PtInSelectedCell(DocPoint,
(LO_CellStruct *)element,
bContinue, start_element,
end_element);
}
else if (element->type != LO_TABLE) {
// Get the rect surrounding selected element,
CRect cRect;
cRect.left = CASTINT(element->x + element->x_offset);
cRect.top = CASTINT(element->y + element->y_offset);
cRect.right = CASTINT(cRect.left + element->width);
cRect.bottom = CASTINT(cRect.top + element->height);
bPtInRegion = cRect.PtInRect( DocPoint );
}
}
// We're done if we are in a rect or finished with last element
if ( bPtInRegion || !bContinue ||
element == (LO_Any_struct*)end_element ) {
break;
}
}
if ( pDrawable ) {
pDrawable->SetOrigin(old_x_origin, old_y_origin);
}
return bPtInRegion;
}
void CWinCX::OnLButtonDownCX(UINT uFlags, CPoint cpPoint)
{
RelayToolTipEvent(cpPoint, WM_LBUTTONDOWN);
// Only do this if clicking is enabled.
if(IsClickingEnabled() == FALSE) {
return;
}
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return;
}
// Start capturing all mouse events.
if(GetPane()) {
::SetCapture(GetPane());
}
XY Point;
ResolvePoint(Point, cpPoint);
#ifdef LAYERS
if (GetContext()->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
event.type = CL_EVENT_MOUSE_BUTTON_DOWN;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.modifiers = (uFlags & MK_SHIFT ? EVENT_SHIFT_MASK : 0)
| (uFlags & MK_CONTROL ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.x = Point.x;
event.y = Point.y;
event.which = 1;
CL_DispatchEvent(GetContext()->compositor,
&event);
}
else
OnLButtonDownForLayerCX(uFlags, cpPoint, Point, NULL);
// Have the mouse timer handler do some dirty work.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
return;
}
void CWinCX::ResetToolTipImg() {
pLastToolTipImg = 0;
m_pLastToolTipAnchor = NULL;
if (m_ToolTip) {
m_ToolTip->Activate(FALSE);
delete m_ToolTip;
m_ToolTip = 0;
}
}
void CWinCX::RelayToolTipEvent(POINT pt, UINT message)
{
if (m_ToolTip) {
MSG msg;
msg.message = message;
msg.hwnd = GetPane();
msg.pt = pt;
::ClientToScreen(msg.hwnd, &msg.pt);
m_ToolTip->RelayEvent(&msg);
}
}
void
CWinCX::OnLButtonDownForLayerCX(UINT uFlags, CPoint &cpPoint, XY& Point,
CL_Layer *layer)
{
MWContext *pMWContext = GetContext();
XP_ASSERT(pMWContext);
// With LAYERS turned on, the orginal method
// OnLButtonDownCX is separated into two methods,
// one of which is a per-layer method.
if (pMWContext->compositor)
CL_GrabMouseEvents(pMWContext->compositor, layer);
#endif /* LAYERS */
#ifdef LAYERS
LO_Element *pElement = GetLayoutElement(Point, layer);
#else
LO_Element *pElement = GetLayoutElement(Point);
#endif /* LAYERS */
// Check for form element and send event back to form element's class.
if (pElement != NULL && pElement->type == LO_FORM_ELE &&
(Point.x - pElement->lo_form.x - pElement->lo_form.x_offset < pElement->lo_form.width) &&
(Point.x - pElement->lo_form.x - pElement->lo_form.x_offset > 0) &&
(Point.y - pElement->lo_form.y - pElement->lo_form.y_offset < pElement->lo_form.height) &&
(Point.y - pElement->lo_form.y - pElement->lo_form.y_offset > 0)) {
CFormElement * pFormElement;
CNetscapeButton * pButton;
switch (pElement->lo_form.element_data->type) {
case FORM_TYPE_BUTTON:
case FORM_TYPE_RESET:
case FORM_TYPE_SUBMIT:
case FORM_TYPE_CHECKBOX:
case FORM_TYPE_RADIO:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
((CNetscapeButton *)CButton::FromHandlePermanent(pFormElement->GetRaw()))
->OnButtonEvent(CL_EVENT_MOUSE_BUTTON_DOWN, uFlags, cpPoint);
break;
case FORM_TYPE_FILE:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
pButton = (CNetscapeButton *)pFormElement->GetSecondaryWidget();
if (pButton)
pButton->OnButtonEvent(CL_EVENT_MOUSE_BUTTON_DOWN, uFlags, cpPoint);
break;
default:
break;
}
return;
}
// Wait to set these until we know the event hasn't be cancelled
// by JS or really meant for a form element. Remember....
m_LastMouseEvent = m_LBDown;
m_cpLBDown = cpPoint;
m_uMouseFlags = uFlags;
m_bLBDown = TRUE;
m_bLBUp = FALSE;
// Deactivate OLE embedded items if active.
CGenericView *pView = GetView();
if(pView != NULL && pElement != (LO_Element *)m_pSelected && m_pSelected != NULL) {
OnDeactivateEmbedCX();
}
#ifdef EDITOR
ED_HitType iTableHit = ED_HIT_NONE;
LO_Element * pTableOrCellElement = NULL;
if( EDT_IS_EDITOR(pMWContext) )
{
m_pStartSelectionCell = NULL;
// Check if user pressed down near a sizeable object border
// and start sizing mode if we are
// Note: X, Y values are in Document coordinates
if( !m_bDragging && !EDT_IsSizing(pMWContext) )
{
BOOL bSizeTable = FALSE;
// Left button down test for selection/sizing/dragging
iTableHit = EDT_GetTableHitRegion( pMWContext, Point.x, Point.y, &pTableOrCellElement, (uFlags & MK_CONTROL) );
// Check if we can select or size a Table or Cell,
// but not if Alt key is pressed, ignore the table
// to allow sizing objects tightly surrounded by Cell border
if( GetAsyncKeyState(VK_MENU) >= 0 &&
pTableOrCellElement )
{
// Save cell we are in
if( pTableOrCellElement->type == LO_CELL )
m_pStartSelectionCell = pTableOrCellElement;
if( iTableHit == ED_HIT_SIZE_TABLE_WIDTH || iTableHit == ED_HIT_SIZE_TABLE_HEIGHT ||
iTableHit == ED_HIT_SIZE_COL || iTableHit == ED_HIT_SIZE_ROW ||
iTableHit == ED_HIT_ADD_ROWS || iTableHit == ED_HIT_ADD_COLS )
{
// We are sizing a table, row, or column
pElement = pTableOrCellElement;
bSizeTable = TRUE;
} else if( iTableHit != ED_HIT_NONE && iTableHit != ED_HIT_DRAG_TABLE )
{
m_bSelectingCells = TRUE;
// Mouse is in a selectable region for table, row, column, or cell.
// If Ctrl key is down and cell is selected, it is appended to other table cells selected
// Otherwise, any other cell or table selected is cleared before new element is selected.
// If Shift key is down, then extend selection to new cell
EDT_SelectTableElement(pMWContext, Point.x, Point.y, pTableOrCellElement, iTableHit,
(uFlags & MK_CONTROL), (uFlags & MK_SHIFT));
goto MOUSE_TIMER;
}
}
if( bSizeTable ||
(pElement && EDT_CanSizeObject(pMWContext, pElement, Point.x, Point.y)) )
{
// Flag to override the normal lock when sizing corners
BOOL bLock = !(BOOL)(uFlags & MK_CONTROL);
XP_Rect rect;
if( EDT_StartSizing(pMWContext, pElement, Point.x, Point.y, bLock, &rect) )
{
// Force redraw of table or cells that might have been selected
// else we have NOT conflicts and garbage at overlaps
UpdateWindow(GetPane());
// Save the new rect.
m_rectSizing.left = rect.left;
m_rectSizing.right = rect.right;
m_rectSizing.top = rect.top;
m_rectSizing.bottom = rect.bottom;
// Draw the initial feedback -- similar to when selected
DisplaySelectionFeedback(LO_ELE_SELECTED, m_rectSizing);
goto MOUSE_TIMER;
}
}
}
}
// Drag Copy/Move - do only if we will not be sizing
if( GetView()->IsKindOf(RUNTIME_CLASS(CNetscapeView)) &&
(iTableHit == ED_HIT_DRAG_TABLE || PtInSelectedRegion(cpPoint, TRUE, layer)) )
{
// Setup to possibly do Drag Copy/Move
// Check these flags during mouse move
// and start drag only if mouse moved enough
if( iTableHit == ED_HIT_DRAG_TABLE )
{
// Setup XP data for possible dragging of table or cells
EDT_StartDragTable(pMWContext, Point.x, Point.y);
}
m_bMouseInSelection = TRUE;
} else
#endif // EDITOR
if ( ! m_bDragging ) {
// If the shift key is down, we need to extend the selection.
if( (uFlags & MK_SHIFT) ) {
#ifdef EDITOR
if( EDT_IS_EDITOR(pMWContext) ){
EDT_ExtendSelection(pMWContext, Point.x, Point.y);
}
else
#endif // EDITOR
{
LO_ExtendSelection(GetDocumentContext(), Point.x, Point.y);
}
} else {
// Start a normal selection
#ifdef EDITOR
if( EDT_IS_EDITOR(pMWContext) ) {
EDT_StartSelection(pMWContext, Point.x, Point.y);
}
else
#endif // EDITOR
{
#ifdef LAYERS
LO_StartSelection(GetDocumentContext(), Point.x, Point.y, layer);
#else
LO_StartSelection(GetDocumentContext(), Point.x, Point.y);
#endif /* LAYERS */
}
}
}
// Highlight an anchor if we are over it.
// Save an image element so we can drag into editor
if ( pElement != NULL && pElement->type == LO_IMAGE ) {
m_pLastImageObject = (LO_ImageStruct*)pElement;
}
// DON'T drag links when in editor - just highlight the text in the link
if(!EDT_IS_EDITOR(pMWContext) &&
pElement != NULL && pElement->type == LO_TEXT && pElement->lo_text.anchor_href != NULL) {
m_pLastArmedAnchor = pElement;
LO_HighlightAnchor(GetDocumentContext(), pElement, TRUE);
}
MOUSE_TIMER:
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(pMWContext);
FEU_MouseTimer(&mt);
}
typedef struct click_closure {
char * szRefer;
int x, y;
CWinCX * pWin;
BOOL bCloseOnFail;
LO_Element * pElement;
} click_closure;
static void
MapToAnchorAndTarget(MWContext * context, LO_Element * pElement, int x, int y,
CString& csAnchor, CString& csTarget)
{
switch(pElement->type) {
case LO_TEXT:
if(pElement->lo_text.anchor_href && pElement->lo_text.anchor_href->anchor) {
csAnchor = (char *) pElement->lo_text.anchor_href->anchor;
csTarget = (char *) pElement->lo_text.anchor_href->target;
}
break;
case LO_IMAGE:
// Check for usemaps (client side ismaps).
if(pElement->lo_image.image_attr && pElement->lo_image.image_attr->usemap_name != NULL) {
LO_AnchorData *pAnchorData = LO_MapXYToAreaAnchor(context,
&pElement->lo_image, x, y);
if(pAnchorData != NULL) {
csAnchor = (char *) pAnchorData->anchor;
csTarget = (char *) pAnchorData->target;
}
break;
}
// Check for ismaps, normal image anchors
if(pElement->lo_image.anchor_href != NULL &&
pElement->lo_image.anchor_href->anchor != NULL) {
if(pElement->lo_image.image_attr && pElement->lo_image.image_attr->attrmask & LO_ATTR_ISMAP) {
char * pAnchor = PR_smprintf("%s?%d,%d",
pElement->lo_image.anchor_href->anchor, x, y);
csAnchor = pAnchor;
XP_FREE(pAnchor);
csTarget =(char *)pElement->lo_image.anchor_href->target;
}
else {
csAnchor = (char *)pElement->lo_image.anchor_href->anchor;
csTarget = (char *)pElement->lo_image.anchor_href->target;
}
}
break;
break;
default:
break;
}
}
//
// Mocha has processed a click on an element. If everything is OK
// we now will do the actual load. If libmocha said to not
// load and we created a new window explicitly for the load
// delete the window
//
static void
win_click_callback(MWContext * pContext, LO_Element * pEle, int32 event,
void * pObj, ETEventStatus status)
{
CString csAnchor, csTarget;
// make sure document hasn't gone away
if(status == EVENT_PANIC) {
if (((click_closure *)pObj)->pElement && pEle) {
XP_FREE(pEle);
}
XP_FREE(pObj);
return;
}
// find out who we are
click_closure * pClose = (click_closure *) pObj;
CWinCX * pWin = pClose->pWin;
// Imagemaps send click pretending to be links. Free the link now and
// set the element back to the image it is.
if (pClose->pElement) {
if(pEle)
XP_FREE(pEle);
pEle = pClose->pElement;
}
MapToAnchorAndTarget(pWin->GetContext(), pEle, pClose->x, pClose->y, csAnchor, csTarget);
#ifdef EDITOR
if( EDT_IS_EDITOR(pWin->GetContext()) ){
// Ctrl Click = edit the URL
FE_LoadUrl((char*)LPCSTR(csAnchor), LOAD_URL_COMPOSER);
goto done;
}
#endif // EDITOR
if(status == EVENT_OK) {
CWinCX *pLoader = pWin->DetermineTarget(csTarget);
pLoader->NormalGetUrl(csAnchor, pClose->szRefer, (csTarget.IsEmpty() ? NULL : (LPCSTR)csTarget));
} else {
if(pClose->bCloseOnFail)
// Make it go away if mocha said no.
FE_DestroyWindow(pContext);
}
done:
if(pClose->szRefer)
XP_FREE(pClose->szRefer);
XP_FREE(pClose);
}
//
// The user clicked on a form image. We told mocha about it and now mocha
// is either going to tell us to submit the form or ignore it
//
static void
image_form_click_callback(MWContext * pContext, LO_Element * pElement, int32 event,
void * pObj, ETEventStatus status)
{
// only continue if OK
if(status != EVENT_OK) {
XP_FREE(pObj);
return;
}
click_closure * pClose = (click_closure *) pObj;
CWinCX * pWin = pClose->pWin;
LO_FormSubmitData *pSubmit = LO_SubmitImageForm(pContext,
&pElement->lo_image,
pClose->x,
pClose->y);
if(pSubmit == NULL) {
// Nothing to do.
return;
}
// Have to do a manual load here.
URL_Struct *pUrl = NET_CreateURLStruct((const char *)pSubmit->action, NET_DONT_RELOAD);
NET_AddLOSubmitDataToURLStruct(pSubmit, pUrl);
// Set the referrer here manually too.
if(pClose->szRefer)
pUrl->referer = XP_STRDUP(pClose->szRefer);
else
pUrl->referer = NULL;
// Request.
pWin->GetUrl(pUrl, FO_CACHE_AND_PRESENT);
// Release.
LO_FreeSubmitData(pSubmit);
}
void CWinCX::OnLButtonUpCX(UINT uFlags, CPoint cpPoint, BOOL &bReturnImmediately)
{
RelayToolTipEvent(cpPoint, WM_LBUTTONUP);
// Only do this if clicking is enabled.
if(IsClickingEnabled() == FALSE) {
return;
}
// Release mouse capture.
if(GetPane()) {
::ReleaseCapture();
}
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return;
}
XY Point;
ResolvePoint(Point, cpPoint);
#ifdef LAYERS
if (GetContext()->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
fe_event.pbReturnImmediately = bReturnImmediately;
event.type = CL_EVENT_MOUSE_BUTTON_UP;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.x = Point.x;
event.y = Point.y;
event.modifiers = (uFlags & MK_SHIFT ? EVENT_SHIFT_MASK : 0)
| (uFlags & MK_CONTROL ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.which = 1;
CL_DispatchEvent(GetContext()->compositor,
&event);
bReturnImmediately = fe_event.pbReturnImmediately;
}
else
OnLButtonUpForLayerCX(uFlags, cpPoint, Point, NULL, bReturnImmediately);
// Have the mouse timer handler do some dirty work.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
return;
}
void
CWinCX::OnLButtonUpForLayerCX(UINT uFlags, CPoint& cpPoint, XY& Point,
CL_Layer *layer, BOOL &bReturnImmediately)
{
History_entry *pHist = NULL;
click_closure * pClosure = NULL;
// With LAYERS turned on, the orginal method
// OnLButtonUpCX is separated into two methods,
// one of which is a per-layer method.
if (GetContext()->compositor)
CL_GrabMouseEvents(GetContext()->compositor, NULL);
#endif /* LAYERS */
#ifdef LAYERS
LO_Element *pElement = GetLayoutElement(Point, layer);
#else
LO_Element *pElement = GetLayoutElement(Point);
#endif /* LAYERS */
if (pElement != NULL && pElement->type == LO_FORM_ELE &&
(Point.x - pElement->lo_form.x - pElement->lo_form.x_offset < pElement->lo_form.width) &&
(Point.x - pElement->lo_form.x - pElement->lo_form.x_offset > 0) &&
(Point.y - pElement->lo_form.y - pElement->lo_form.y_offset < pElement->lo_form.height) &&
(Point.y - pElement->lo_form.y - pElement->lo_form.y_offset > 0)) {
CFormElement * pFormElement;
CNetscapeButton * pButton;
switch (pElement->lo_form.element_data->type) {
case FORM_TYPE_BUTTON:
case FORM_TYPE_RESET:
case FORM_TYPE_SUBMIT:
case FORM_TYPE_CHECKBOX:
case FORM_TYPE_RADIO:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
((CNetscapeButton *)CButton::FromHandlePermanent(pFormElement->GetRaw()))
->OnButtonEvent(CL_EVENT_MOUSE_BUTTON_UP, uFlags, cpPoint);
break;
case FORM_TYPE_FILE:
pFormElement=(CFormElement *)pElement->lo_form.element_data->ele_minimal.FE_Data;
pButton = (CNetscapeButton *)pFormElement->GetSecondaryWidget();
if (pButton)
pButton->OnButtonEvent(CL_EVENT_MOUSE_BUTTON_UP, uFlags, cpPoint);
break;
default:
break;
}
m_bLBDown = FALSE;
m_bLBUp = TRUE;
return;
}
// Wait to set these until we know the event hasn't be cancelled
// by JS or really meant for a form element. Remember....
m_LastMouseEvent = m_LBUp;
m_cpLBUp = cpPoint;
m_uMouseFlags = uFlags;
m_bLBDown = FALSE;
m_bLBUp = TRUE;
// JS needs screen coords for click events.
CPoint cpScreenPoint(cpPoint);
ClientToScreen(GetPane(), &cpScreenPoint);
#ifdef EDITOR
m_pStartSelectionCell = NULL;
MWContext *pMWContext = GetContext();
// Finish sizing an object only if we moved enough
if( EDT_IS_SIZING ){
// Remove last sizing feedback
DisplaySelectionFeedback(LO_ELE_SELECTED, m_rectSizing);
// We need this check or it is impossible to place a caret between
// adjacent objects)
if(abs(m_cpLBUp.x - m_cpLBDown.x) > CLICK_THRESHOLD ||
abs(m_cpLBUp.y - m_cpLBDown.y) > CLICK_THRESHOLD) {
ASSERT(cpPoint.y == Point.y - GetOriginY()); // REMOVEME
// Resize the object
EDT_EndSizing(pMWContext);
// See comment at end - shouldn't return here???
// (many other places below do return!)
goto MOUSE_TIMER;
} else {
EDT_CancelSizing(pMWContext);
// Move caret to where cursor is
EDT_StartSelection(pMWContext, Point.x, Point.y);
}
} else if(!m_bDragging && !m_bSelectingCells) {
// If within an existing selection, we didn't trigger StartSelection
// on MouseDown, so do it now
if( m_bMouseInSelection ){
if( EDT_IS_EDITOR(pMWContext) ) {
EDT_StartSelection(pMWContext, Point.x, Point.y);
} else {
#ifdef LAYERS
LO_StartSelection(GetDocumentContext(), Point.x, Point.y, layer);
#else
LO_StartSelection(GetDocumentContext(), Point.x, Point.y);
#endif /* LAYERS */
}
}
// Cleanup selection - check if start=end
if( EDT_IS_EDITOR(pMWContext) )
EDT_EndSelection(pMWContext, Point.x, Point.y);
}
if( EDT_IS_EDITOR(pMWContext) )
{
if( EDT_CanPasteStyle(pMWContext) )
{
// Apply style if available
EDT_PasteStyle(pMWContext, TRUE);
}
// Cancel any attempt to drag a table or cell
EDT_StopDragTable(pMWContext);
}
#endif // EDITOR
m_bMouseInSelection = FALSE;
m_bDragging = FALSE;
m_bSelectingCells = FALSE;
// If we previously highlighted an anchor in button down, unhighlight it.
if(m_pLastArmedAnchor != NULL) {
LO_HighlightAnchor(GetDocumentContext(), m_pLastArmedAnchor, FALSE);
m_pLastArmedAnchor = NULL;
}
// If the user moved the mouse beyond the clicking threshold, then consider
// this a text selection operation and don't continue.
if(abs(m_cpLBUp.x - m_cpLBDown.x) > CLICK_THRESHOLD ||
abs(m_cpLBUp.y - m_cpLBDown.y) > CLICK_THRESHOLD) {
return;
}
#ifdef EDITOR
// Done with possible drag image
m_pLastImageObject = NULL;
#endif
// If there's no element, there's no need to see if we should load something.
// Just send a JS Click event to the document and return.
if(pElement == NULL) {
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_CLICK;
event->which = 1;
event->x = Point.x;
event->y = Point.y;
event->docx = Point.x + CL_GetLayerXOrigin(layer);
event->docy = Point.y + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), NULL, event, NULL, NULL);
return;
}
//
// Control click to chase anchors in the editor
//
#ifdef EDITOR
if((uFlags & MK_CONTROL) == 0 && EDT_IS_EDITOR(GetContext())){
m_pLastImageObject = NULL;
return;
}
#endif
// It's probably we're loading.
// Figure out what we'll be sending as the referrer field.
pHist = SHIST_GetCurrent(&GetContext()->hist);
pClosure = XP_NEW_ZAP(click_closure);
pClosure->pWin = this;
pClosure->pElement = NULL;
if(pHist != NULL && pHist->address != NULL)
pClosure->szRefer = strdup(pHist->origin_url ? pHist->origin_url : pHist->address);
else
pClosure->szRefer = NULL;
// To find out what to do, switch on the element's type.
switch(pElement->type)
{
case LO_TEXT:
if(pElement->lo_text.anchor_href && pElement->lo_text.anchor_href->anchor)
{
// set tab_focus to this text element with a link.
//#52932 setFormElementTabFocus( (LO_Element *) pElement );
// if the shift-key is down save the object --- don't load it
if(uFlags & MK_SHIFT && NOT_A_DIALOG(this))
{
// Should Mocha block save operations too?
CSaveCX::SaveAnchorObject((const char *)pElement->lo_text.anchor_href->anchor, NULL);
XP_FREE(pClosure);
return;
}
// we are about to follow a link via a user click --
// tell the mocha library. Don't follow the link
// until we get to our closure
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_CLICK;
event->which = 1;
event->x = Point.x;
event->y = Point.y;
event->docx = Point.x + CL_GetLayerXOrigin(layer);
event->docy = Point.y + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), pElement, event, win_click_callback, pClosure);
return;
}
break;
case LO_IMAGE:
{
// Glean the FE data.
// Figure out where the click occurred.
LO_ImageStruct* pImage = &pElement->lo_image;
// Layout wants this in pixels, not FE units.
// Point is in layer coordinates, as is the position of the image
CPoint cpMap((int) (Point.x - Twips2PixX(pImage->x + pImage->x_offset + pImage->border_width)),
(int) (Point.y - Twips2PixY(pImage->y + pImage->y_offset + pImage->border_width)));
pClosure->x = cpMap.x;
pClosure->y = cpMap.y;
// Check for usemaps (client side ismaps).
if(pElement->lo_image.image_attr->usemap_name != NULL)
{
LO_AnchorData *pAnchorData = LO_MapXYToAreaAnchor(GetDocumentContext(),
&pElement->lo_image, cpMap.x, cpMap.y);
if(pAnchorData != NULL)
{
// set tab_focus to this image element with a link. todo need to get the area index
//setFormElementTabFocus( (LO_Element *) pElement );
if(uFlags & MK_SHIFT && NOT_A_DIALOG(this)) {
// Save it.
CSaveCX::SaveAnchorObject((const char *)pAnchorData->anchor, NULL);
XP_FREE(pClosure);
}
else {
LO_Element * pDummy;
// Imagemap area pretend to be links for JavaScript.
pDummy = (LO_Element *) XP_NEW_ZAP(LO_Element);
pDummy->lo_text.type = LO_TEXT;
pDummy->lo_text.anchor_href = pAnchorData;
// We use the text of the element to determine if it is still
// valid later so give the dummy text struct's text a value.
if (pDummy->lo_text.anchor_href->anchor)
pDummy->lo_text.text = pDummy->lo_text.anchor_href->anchor;
// We'll need the image element later in the callback.
pClosure->pElement = pElement;
// we are about to follow a link via a user click --
// tell the mocha library. Don't follow the link
// until we get to our closure
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_CLICK;
event->which = 1;
event->x = Point.x;
event->y = Point.y;
event->docx = Point.x + CL_GetLayerXOrigin(layer);
event->docy = Point.y + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), pDummy, event,
win_click_callback, pClosure);
}
// We're out, don't fall through.
return;
}
}
// Check for ismaps, normal image anchors
else if(pElement->lo_image.anchor_href != NULL &&
pElement->lo_image.anchor_href->anchor != NULL)
{
// set tab_focus to this image element with a link.
//#52932 setFormElementTabFocus( (LO_Element *) pElement );
// Ismap?
if(pElement->lo_image.image_attr->attrmask & LO_ATTR_ISMAP)
{
if(uFlags & MK_SHIFT && NOT_A_DIALOG(this))
{
char * pAnchor = PR_smprintf("%s?%d,%d",
pElement->lo_image.anchor_href->anchor, cpMap.x, cpMap.y);
CSaveCX::SaveAnchorObject(pAnchor, NULL);
XP_FREE(pClosure);
XP_FREE(pAnchor);
}
else {
// set tab_focus to this image element with a link.
//#52932 setFormElementTabFocus((LO_Element *) pElement);
// remember where we are going
pClosure->x = cpMap.x;
pClosure->y = cpMap.y;
// we are about to follow a link via a user click --
// tell the mocha library. Don't follow the link
// until we get to our closure
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_CLICK;
event->which = 1;
event->x = Point.x;
event->y = Point.y;
event->docx = Point.x + CL_GetLayerXOrigin(layer);
event->docy = Point.y + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), pElement, event,
win_click_callback, pClosure);
}
}
// Anchor.
else {
if(uFlags & MK_SHIFT && NOT_A_DIALOG(this)) {
// Should mocha block save operations too?
CSaveCX::SaveAnchorObject((const char *)pElement->lo_image.anchor_href->anchor, NULL);
XP_FREE(pClosure);
}
else {
// we are about to follow a link via a user click --
// tell the mocha library. Don't follow the link
// until we get to our closure
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_CLICK;
event->which = 1;
event->x = Point.x;
event->y = Point.y;
event->docx = Point.x + CL_GetLayerXOrigin(layer);
event->docy = Point.y + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), pElement, event,
win_click_callback, pClosure);
}
}
// We're out, don't fall through.
return;
}
// Check for images as submit buttons.
// Lot's of these on test pages, but in the real world?
if(pElement->lo_image.image_attr &&
(pElement->lo_image.image_attr->attrmask & LO_ATTR_ISFORM))
{
pClosure->x = cpMap.x;
pClosure->y = cpMap.y;
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_SUBMIT;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), (LO_Element *)&pElement->lo_image, event,
image_form_click_callback, pClosure);
return;
}
break; // case LO_IMAGE
}
default:
// nothing doing.
XP_FREE(pClosure);
break;
}
MOUSE_TIMER:
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
}
// convert LO_Text to a bookmark object we can drag into
// the bookmark window
PRIVATE HGLOBAL
wfe_textObjectToBookmarkHandle(LO_TextStruct * text, char * title)
{
if(!text)
return(NULL);
HGLOBAL hBookmark = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT, sizeof(BOOKMARKITEM));
if(!hBookmark)
return(NULL);
LPBOOKMARKITEM pBookmark = (LPBOOKMARKITEM)GlobalLock(hBookmark);
if(text->anchor_href && text->anchor_href->anchor)
PR_snprintf(pBookmark->szAnchor,sizeof(pBookmark->szAnchor),"%s",text->anchor_href->anchor);
if( title ) {
// First try to use the title supplied,
// This may contain more text than the text structure has
PR_snprintf(pBookmark->szText,sizeof(pBookmark->szText),"%s",title);
} else if(text->text) {
PR_snprintf(pBookmark->szText,sizeof(pBookmark->szText),"%s",text->text);
}
GlobalUnlock(hBookmark);
return hBookmark;
}
// Creates OLE drag data source for selected text
void CWinCX::DragSelection()
{
// Begin the drag and drop operation
// REMEMBER: OnDrop: Check if end pt is withing selection,
// if yes, ignore drop
MWContext * pMWContext = GetContext();
#ifdef EDITOR
// Don't bother if no selection or not allowed in Editor
if ( EDT_IS_EDITOR(pMWContext) &&
EDT_COP_OK != EDT_CanCopy(pMWContext, TRUE))
return;
#endif // EDITOR
// Here's where we put the data
COleDataSource * pDataSource = new COleDataSource;
// This is used to override cursors during dragging
UINT nDragType = FE_DRAG_TEXT;
#ifdef EDITOR
if( EDT_IS_EDITOR(pMWContext) )
{
nDragType = EDT_IsDraggingTable(pMWContext) ? FE_DRAG_TABLE : FE_DRAG_HTML;
}
#endif
CViewDropSource * pDropSource = new CViewDropSource(nDragType);
char* pText = NULL;
XP_HUGE_CHAR_PTR pGlobal;
XP_HUGE_CHAR_PTR pHTML;
int32 textLen = 0;
int32 htmlLen;
m_bDragging = FALSE;
#ifdef EDITOR
if( EDT_IS_EDITOR(pMWContext) ){
if( EDT_COP_OK == EDT_CanCopy(pMWContext, TRUE) &&
EDT_COP_OK == EDT_CopySelection(pMWContext, &pText, &textLen, &pHTML, &htmlLen) ){
// Put HTML-formated text in OLE data object
HGLOBAL hHTML = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT, (int)htmlLen);
if(hHTML) {
pGlobal = (char *) GlobalLock(hHTML);
XP_HUGE_MEMCPY(pGlobal, pHTML, (int) htmlLen);
XP_HUGE_FREE(pHTML);
GlobalUnlock(hHTML);
pDataSource->CacheGlobalData(RegisterClipboardFormat(NETSCAPE_EDIT_FORMAT), hHTML);
m_bDragging = TRUE;
}
}
}
else
#endif // EDITOR
{
// Browser-only
pText = (char *) LO_GetSelectionText(GetDocumentContext());
if( pText )
textLen = XP_STRLEN(pText);
}
// Put unformated text in OLE data object - may be dropped in other containers
HGLOBAL hText = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT, (int)textLen+1);
DROPEFFECT res = 0;
if( pText && textLen > 0 ){
#ifdef XP_WIN32
// Also try to put CF_UNICODETEXT
int datacsid = INTL_GetCSIWinCSID(LO_GetDocumentCharacterSetInfo( GetDocumentContext() )) & ~CS_AUTO;
if((CS_USER_DEFINED_ENCODING != datacsid) && (0 != datacsid))
{
int len = (INTL_StrToUnicodeLen(datacsid, (unsigned char*)pText)+1) * 2;
HGLOBAL hData = GlobalAlloc(GMEM_MOVEABLE, (DWORD) len);
if(hData)
{
unsigned char* string = (unsigned char *) GlobalLock(hData);
if(string)
{
INTL_StrToUnicode(datacsid, (unsigned char*)pText, (INTL_Unicode*)string, len);
GlobalUnlock(hData);
pDataSource->CacheGlobalData(CF_UNICODETEXT, hData);
}
}
}
#endif
if(hText) {
pGlobal = (char *) GlobalLock(hText);
XP_MEMCPY(pGlobal, pText, (int) textLen+1);
XP_FREE(pText);
GlobalUnlock(hText);
pDataSource->CacheGlobalData(CF_TEXT, hText);
m_bDragging = TRUE;
}
}
if( m_bDragging ){
BOOL bWaitingMode = pMWContext->waitingMode;
// Prevent closing/interaction with source window while dragging
pMWContext->waitingMode = TRUE;
res = pDataSource->DoDragDrop(DROPEFFECT_COPY | DROPEFFECT_MOVE | DROPEFFECT_SCROLL,
NULL, pDropSource);
pMWContext->waitingMode = bWaitingMode;
}
m_bDragging = FALSE;
#ifdef EDITOR
if( EDT_IS_EDITOR(pMWContext) ){
EDT_StopDragTable(pMWContext);
}
#endif
// Prevent selection-extension when finished
m_bLBDown = FALSE;
m_bLBUp = TRUE;
// its over so clean up
pDataSource->Empty();
delete pDataSource;
delete pDropSource;
}
// Triggered on button up on our bitmap on the menu
void CWinCX::CopyCurrentURL()
{
MWContext *pMWContext = GetContext();
if ( pMWContext == NULL ) {
return;
}
History_entry * hist_ent = SHIST_GetCurrent(&(pMWContext->hist));
if ( hist_ent == NULL || hist_ent->address == NULL ){
return;
}
HGLOBAL hData = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT,
XP_STRLEN((char *) hist_ent->address) + 2);
if(!hData) {
return;
}
// lock the string and copy the data over
char * pString = (char *) GlobalLock(hData);
strcpy(pString, (char *) hist_ent->address);
GlobalUnlock(hData);
GetFrame()->GetFrameWnd()->OpenClipboard();
::EmptyClipboard();
::SetClipboardData(CF_TEXT, hData);
// Also copy bookmark-formatted data so we can paste full link,
// not just text, into the Editor
hData = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT, sizeof(BOOKMARKITEM));
if(hData){
LPBOOKMARKITEM pBookmark = (LPBOOKMARKITEM)GlobalLock(hData);
PR_snprintf(pBookmark->szAnchor, sizeof(pBookmark->szAnchor), "%s",
hist_ent->address);
PR_snprintf(pBookmark->szText, sizeof(pBookmark->szText), "%s",
hist_ent->title);
GlobalUnlock(hData);
::SetClipboardData(RegisterClipboardFormat(NETSCAPE_BOOKMARK_FORMAT), hData);
}
::CloseClipboard();
}
// Drag from the Bitmap Menu item or URL bar icon
void CWinCX::DragCurrentURL()
{
MWContext *pMWContext = GetContext();
if ( pMWContext == NULL ) {
return;
}
History_entry * hist_ent = SHIST_GetCurrent(&(pMWContext->hist));
if ( hist_ent == NULL || hist_ent->address == NULL ){
return;
}
HGLOBAL hAddrString = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT,
XP_STRLEN((char *) hist_ent->address) + 2);
if(!hAddrString) {
return;
}
// lock the string and copy the data over
char * pAddrString = (char *) GlobalLock(hAddrString);
strcpy(pAddrString, (char *) hist_ent->address);
GlobalUnlock(hAddrString);
HGLOBAL hBookmark = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT, sizeof(BOOKMARKITEM));
if(!hBookmark) {
return;
}
LPBOOKMARKITEM pBookmark = (LPBOOKMARKITEM)GlobalLock(hBookmark);
PR_snprintf(pBookmark->szAnchor,sizeof(pBookmark->szAnchor),"%s",hist_ent->address);
PR_snprintf(pBookmark->szText,sizeof(pBookmark->szText),"%s",hist_ent->title);
GlobalUnlock(hBookmark);
// Create the DataSourceObject
COleDataSource * pDataSource = new COleDataSource;
pDataSource->CacheGlobalData(CF_TEXT, hAddrString);
pDataSource->CacheGlobalData(RegisterClipboardFormat(NETSCAPE_BOOKMARK_FORMAT), hBookmark);
// This is used to override cursors during dragging
CViewDropSource * pDropSource = new CViewDropSource(FE_DRAG_LINK);
DragInternetShortcut ( pDataSource,
(char*)hist_ent->title,
(char*)hist_ent->address );
// do the drag/drop operation.
// set the m_bDragging flag so that we can prevent ourseleves from dropping on
// ourselves
m_bDragging = TRUE;
// no saved image for next time
// Must do this before DoDragDrop to prevent
// arriving here again!
m_pLastImageObject = NULL;
m_pLastArmedAnchor = NULL;
// Don't start drag until outside this rect
RECT rectDragStart = {0,0,20,20};
// We supply the DropSource object instead of default behavior
// This prevents closing source frame during drag and drop
BOOL bWaitingMode = pMWContext->waitingMode;
pMWContext->waitingMode = TRUE;
pDataSource->DoDragDrop(DROPEFFECT_COPY | DROPEFFECT_LINK | DROPEFFECT_MOVE | DROPEFFECT_SCROLL,
&rectDragStart, pDropSource);
pMWContext->waitingMode = bWaitingMode;
m_bDragging = FALSE;
// After dragging, moving mouse in browser acts like button is down,
// and ugly selection extension happens. This prevents that.
m_bLBDown = FALSE;
m_bLBUp = TRUE;
// its over so clean up
pDataSource->Empty();
delete pDataSource;
delete pDropSource;
}
void wfe_Progress(MWContext *pContext, const char *pMessage);
void CWinCX::OnMouseMoveCX(UINT uFlags, CPoint cpPoint, BOOL &bReturnImmediately)
{
// Must have a view to continue.
if(GetPane() == NULL) {
return;
}
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return;
}
// This is set TRUE only by CNetscapeEditView::OnRButtonDown
// while popup menu is active
// We ignore this message else it changes cursor to something inappropriate
if( m_bInPopupMenu )
return;
// Remember....
m_LastMouseEvent = m_MMove;
m_cpMMove = cpPoint;
m_uMouseFlags = uFlags;
// Convert from screen to window coordinates
XY xyPoint;
ResolvePoint(xyPoint, cpPoint);
#ifdef LAYERS
MWContext * context = GetContext();
if (context->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
fe_event.pbReturnImmediately = bReturnImmediately;
event.type = CL_EVENT_MOUSE_MOVE;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.x = xyPoint.x;
event.y = xyPoint.y;
event.which = 1;
event.modifiers = 0;
CL_DispatchEvent(context->compositor, &event);
}
else
OnMouseMoveForLayerCX(uFlags, cpPoint, xyPoint, NULL, bReturnImmediately);
// Have the mouse timer handler do some dirty work.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
return;
}
void
CWinCX::OnMouseMoveForLayerCX(UINT uFlags, CPoint& cpPoint,
XY& xyPoint, CL_Layer *layer, BOOL &bReturnImmediately)
{
// With LAYERS turned on, the orginal method
// OnMouseMoveCX is separated into two methods,
// one of which is a per-layer method.
#endif /* LAYERS */
MWContext * context = GetContext();
BOOL bTextSet = FALSE;
LO_Element *pElement = GetLayoutElement(xyPoint, layer);
if (pElement && (pElement->type == LO_IMAGE)) {
LO_ImageStruct* pImage = &pElement->lo_image;
if (pImage) {
CreateToolTip(pImage, cpPoint, layer);
RelayToolTipEvent(cpPoint, WM_MOUSEMOVE);
}
}
int32 xVal = xyPoint.x;
int32 yVal = xyPoint.y;
// don't do anything if we are waiting for the netlib to get
// into gear
if (context->waitingMode) {
// Change cursor only if not doing internal drag
if( !m_bDragging ){
SetCursor(theApp.LoadStandardCursor(IDC_WAIT));
}
return; // can't load while loading
}
// Note: don't use (uFlags & MK_LBUTTON) because this is
// reported as TRUE on a mouse move message
// following a click off of a dialog or popup menu
//
if ( m_bLBDown )
{
#ifdef EDITOR
// If we are selecting cells, skip the rest
if( m_bSelectingCells )
{
// We must fire an event to setup CLOSURE struct
// and call mouse_closure_callback
FireMouseOverEvent(NULL, xVal, yVal, layer);
goto MOUSE_TIMER;
}
if( EDT_IS_SIZING )
{
// We are sizing
BOOL bLock = !(BOOL)(uFlags & MK_CONTROL);
XP_Rect new_rect;
if( EDT_GetSizingRect(context, xVal, yVal, bLock, &new_rect) )
{
// Remove last sizing feedback
DisplaySelectionFeedback(LO_ELE_SELECTED, m_rectSizing);
// Save the new rect.
m_rectSizing.left = new_rect.left;
m_rectSizing.right = new_rect.right;
m_rectSizing.top = new_rect.top;
m_rectSizing.bottom = new_rect.bottom;
// then draw new feedback
DisplaySelectionFeedback(LO_ELE_SELECTED, m_rectSizing);
}
// Status text was set by XP code, so set flag here
bTextSet = TRUE;
goto MOUSE_TIMER;
}
// Check for cell selection only if not starting drag of table cells
if( !EDT_IsDraggingTable(context) )
{
// We are not currently selecting cells, so get the cell we may be over
// Note: This will return cell and ED_HIT_SIZE_COL if inbetween columns,
// so check that because we don't want to select cell if just before the left edge
LO_Element *pCellElement = NULL;
// Mouse move test with left button down - extend selection to other cells
ED_HitType iTableHit = EDT_GetTableHitRegion(context, xVal, yVal, &pCellElement, FALSE);
if( m_pStartSelectionCell && pCellElement &&
(iTableHit != ED_HIT_SIZE_COL) &&
pCellElement->type == LO_CELL && (pCellElement != m_pStartSelectionCell) )
{
m_bSelectingCells = TRUE;
if( m_pStartSelectionCell )
{
// Mouse is in a different cell then when we started selecting
// So switch to cell-selection mode.
// 1st FALSE param means clear any other selection (shouldn't be any)
// (last param is used to extend selection)
EDT_SelectTableElement(context, m_pStartSelectionCell->lo_any.x, m_pStartSelectionCell->lo_any.y,
m_pStartSelectionCell, ED_HIT_SEL_CELL, FALSE, FALSE);
}
// Select new cell as well: If previously selecting, last param = TRUE
// and we append this cell
EDT_SelectTableElement(context, xVal, yVal, pCellElement, ED_HIT_SEL_CELL,
FALSE, m_pStartSelectionCell != NULL);
goto MOUSE_TIMER;
}
}
#endif // EDITOR
// Don't bother to do selection or dragging unless we actually moved
if( (abs(cpPoint.x - m_cpLBDown.x) > 5)
|| (abs(cpPoint.y - m_cpLBDown.y) > 5) )
{
if( m_bMouseInSelection )
{
// release the mouse capture
::ReleaseCapture();
m_bMouseInSelection = FALSE;
// Get and drag the selection
DragSelection();
// Don't do anything else if we are dragging!
goto MOUSE_TIMER;
}
// Extend the selection
#ifdef EDITOR
if( EDT_IS_EDITOR(context) )
{
EDT_ExtendSelection(context, xVal, yVal);
} else
#endif // EDITOR
{
LO_ExtendSelection(GetDocumentContext(), xVal, yVal);
}
}
int32 lYPos = GetOriginY();
int32 lXPos = GetOriginX();
int32 xCur = xVal;
int32 yCur = yVal;
#ifdef LAYERS
if (layer)
{
int32 layer_x_offset = CL_GetLayerXOrigin(layer);
int32 layer_y_offset = CL_GetLayerYOrigin(layer);
xCur += layer_x_offset;
yCur += layer_y_offset;
}
#endif // LAYERS
if(xCur < GetOriginX()) {
lXPos = xCur;
}
else if(xCur > GetWidth() + GetOriginX()) {
lXPos = xCur - GetWidth();
}
if(yCur < GetOriginY()) {
lYPos = yCur;
}
else if(yVal > GetHeight() + GetOriginY()) {
lYPos = yCur - GetHeight();
}
// Validate position recommendations, and reposition if necessary.
if(lXPos > GetDocumentWidth() - GetWidth()) {
lXPos = GetDocumentWidth() - GetWidth();
}
if(lXPos < 0) {
lXPos = GetOriginX();
}
if(lYPos > GetDocumentHeight() - GetHeight()) {
lYPos = GetDocumentHeight() - GetHeight();
}
if(lYPos < 0) {
lYPos = GetOriginY();
}
if(lYPos != GetOriginY() || lXPos != GetOriginX()) {
// Reposition.
SetDocPosition(context, FE_VIEW, lXPos, lYPos);
}
// We need to unhighlight an anchor if we highlighted it on buttonDown
// The anchor element is held within the last_armed_xref global
// If last_armed_xref is non-null then the anchor is highlighted and
// needs to be unhighlighted and a drag and drop operation started.
if( m_pLastArmedAnchor && !EDT_IS_EDITOR(context) ) {
if(abs(m_cpMMove.x - m_cpLBDown.x) > CLICK_THRESHOLD ||
abs(m_cpMMove.y - m_cpLBDown.y) > CLICK_THRESHOLD) {
LO_HighlightAnchor(GetDocumentContext(), m_pLastArmedAnchor, FALSE);
// Convert the selection into stuff other people can understand
LO_TextStruct * text = (LO_TextStruct *) m_pLastArmedAnchor;
if(!text)
goto MOUSE_TIMER;
int len = XP_STRLEN((char *) text->anchor_href->anchor) + 2;
// make an assumption that will bite us later. Shove everything into
// global space cuz it will be small
HGLOBAL hAddrString = GlobalAlloc(GMEM_DDESHARE | GMEM_ZEROINIT, len);
if(!hAddrString)
goto MOUSE_TIMER;
// lock the string and copy the data over
char * pAddrString = (char *) GlobalLock(hAddrString);
strcpy(pAddrString, (char *) text->anchor_href->anchor);
GlobalUnlock(hAddrString);
//CLM: Select the full link object, which scans neighboring text elements
// to gather text that may have different formatting but same HREF
XY Point;
ResolvePoint(Point, m_cpLBDown);
#ifdef LAYERS
LO_SelectObject(GetDocumentContext(), Point.x, Point.y, NULL);
#else
LO_SelectObject(GetDocumentContext(), Point.x, Point.y);
#endif /* LAYERS */
// check to see if its an address book url. If it is convert it to
// vcard clipboard format
COleDataSource * pDataSource = NULL;
CViewDropSource * pDropSource = NULL;
char * url = (char*) text->anchor_href->anchor;
char * path = NET_ParseURL((char *) text->anchor_href->anchor, GET_PATH_PART);
char * search = NET_ParseURL((char *) text->anchor_href->anchor, GET_SEARCH_PART);
if (!XP_STRNCASECMP(path,"add",3)) {
if (!XP_STRNCASECMP (search, "?vcard=", 7)) {
// Create the DataSourceObject
CLIPFORMAT mVcardClipboardFormat = (CLIPFORMAT)RegisterClipboardFormat(vCardClipboardFormat);
char * escVcard = XP_STRDUP (search+7);
if (escVcard) {
escVcard = NET_UnEscape(escVcard);
HANDLE hString = 0;
hString = GlobalAlloc(GMEM_MOVEABLE|GMEM_DDESHARE|GMEM_ZEROINIT,strlen(escVcard)+1);
LPSTR lpszString = (LPSTR)GlobalLock(hString);
strcpy(lpszString, escVcard);
GlobalUnlock(hString);
XP_FREEIF (escVcard);
pDataSource = new COleDataSource;
pDataSource->CacheGlobalData(mVcardClipboardFormat, hString);
pDataSource->CacheGlobalData(CF_TEXT, hString);
pDropSource = new CViewDropSource(FE_DRAG_VCARD);
}
}
}
else {
char *pFullLink = (char *) LO_GetSelectionText(GetDocumentContext());
HGLOBAL hBookmark = wfe_textObjectToBookmarkHandle(text, pFullLink);
// make sure we have a bookmark format defined
CLIPFORMAT mBookmarkClipboardFormat = (CLIPFORMAT)RegisterClipboardFormat(NETSCAPE_BOOKMARK_FORMAT);
// Create the DataSourceObject
pDataSource = new COleDataSource;
pDataSource->CacheGlobalData(CF_TEXT, hAddrString);
pDataSource->CacheGlobalData(mBookmarkClipboardFormat, hBookmark);
// This is used to override cursors during dragging
pDropSource = new CViewDropSource(FE_DRAG_LINK);
DragInternetShortcut ( pDataSource,
pFullLink ? pFullLink : (char*)text->text,
(char*)text->anchor_href->anchor );
}
// do the drag/drop operation.
// set the m_bDragging flag so that we can prevent ourseleves from dropping on
// ourselves
m_bDragging = TRUE;
// Must do this before DoDragDrop to prevent
// arriving here again!
m_pLastArmedAnchor = NULL;
m_pLastImageObject = NULL;
// This prevents closing source frame during drag and drop
BOOL bWaitingMode = context->waitingMode;
context->waitingMode = TRUE;
// We supply the DropSource object instead of default behavior
// (we don't need return value, do we?)
pDataSource->DoDragDrop(DROPEFFECT_COPY | DROPEFFECT_LINK | DROPEFFECT_MOVE | DROPEFFECT_SCROLL,
NULL, pDropSource);
context->waitingMode = bWaitingMode;
m_bDragging = FALSE;
// After dragging, moving mouse in browser acts like button is down,
// and ugly selection extension happens. This prevents that.
m_bLBDown = FALSE;
m_bLBUp = TRUE;
// its over so clean up
pDataSource->Empty();
delete pDataSource;
delete pDropSource;
}
}
#ifdef EDITOR
// Drag image from an Editor or Browser
else if ( m_pLastImageObject &&
(abs(m_cpMMove.x - m_cpLBDown.x) > CLICK_THRESHOLD ||
abs(m_cpMMove.y - m_cpLBDown.y) > CLICK_THRESHOLD) ) {
LO_ImageStruct *pImage = (LO_ImageStruct *)m_pLastImageObject;
char *pImageURL = NULL;
HGLOBAL hImageData = WFE_CreateCopyImageData(context, pImage);
if ( hImageData ) {
// make sure we have a clipboard format defined
CLIPFORMAT mImageFormat = (CLIPFORMAT)RegisterClipboardFormat(
NETSCAPE_IMAGE_FORMAT);
// Create the DataSourceObject
COleDataSource * pDataSource = new COleDataSource;
pDataSource->CacheGlobalData(mImageFormat, hImageData);
// This is used to override cursors during dragging
CViewDropSource * pDropSource = new CViewDropSource(FE_DRAG_IMAGE);
// do the drag/drop operation.
// set the m_bDragging flag so that we can prevent ourseleves from dropping on
// ourselves
m_bDragging = TRUE;
// no saved image for next time
m_pLastImageObject = NULL;
m_pLastArmedAnchor = NULL;
//Weird problem:
// Return from dropping after InsertImage
// sometimes results in hourglass cursor
// So save and restore after DoDragDrop
HCURSOR hCursor = GetCursor();
// This prevents closing source frame during drag and drop
BOOL bWaitingMode = context->waitingMode;
context->waitingMode = TRUE;
pDataSource->DoDragDrop(DROPEFFECT_COPY | DROPEFFECT_MOVE | DROPEFFECT_SCROLL,
NULL, pDropSource);
context->waitingMode = bWaitingMode;
SetCursor(hCursor);
m_bDragging = FALSE;
// Prevent selection mode when returning to browser
m_bLBDown = FALSE;
m_bLBUp = TRUE;
// its over so clean up
pDataSource->Empty();
delete pDataSource;
delete pDropSource;
}
}
#endif // EDITOR
}
else { // Left button is not down...
// If there are connections being initiated (i.e. the watch
// cursor is up) don't blow away the text so that the netlib
// messages persist in the status bar
if(context->waitingMode)
goto MOUSE_TIMER;
#ifdef LAYERS
LO_Element *lo_element = LO_XYToElement(GetDocumentContext(), xVal, yVal, layer);
#else
LO_Element *lo_element = LO_XYToElement(GetDocumentContext(), xVal, yVal);
#endif /* LAYERS */
// Handle mouse over event processing with back end libs.
if(!bTextSet) {
// Let the backend take a crack at handling the text
FireMouseOverEvent(lo_element,xVal, yVal, layer);
// if the backend didn't set the text we will set the
// text in our closure --- don't do anything else
// in this routine
}
}
MOUSE_TIMER:
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(context);
FEU_MouseTimer(&mt);
}
void CWinCX::OnRButtonDblClkCX(UINT uFlags, CPoint cpPoint) {
// Only do this if clicking is enabled.
if(IsClickingEnabled() == FALSE) {
return;
}
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return;
}
XY xyPoint;
ResolvePoint(xyPoint, cpPoint);
if (GetContext()->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
event.type = CL_EVENT_MOUSE_BUTTON_MULTI_CLICK;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.x = xyPoint.x;
event.y = xyPoint.y;
event.modifiers = (uFlags & MK_SHIFT ? EVENT_SHIFT_MASK : 0)
| (uFlags & MK_CONTROL ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.which = 3;
event.data = 2;
CL_DispatchEvent(GetContext()->compositor, &event);
}
else
OnRButtonDblClkForLayerCX(uFlags, cpPoint, xyPoint, NULL);
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
}
void
CWinCX::OnRButtonDblClkForLayerCX(UINT uFlags, CPoint& cpPoint,
XY& Point, CL_Layer *layer)
{
// With LAYERS turned on, the orginal method
// OnRButtonDblClkCX is separated into two methods,
// one of which is a per-layer method.
// Remember....
m_LastMouseEvent = m_RBDClick;
m_cpRBDClick = cpPoint;
m_uMouseFlags = uFlags;
}
void CWinCX::OnRButtonDownCX(UINT uFlags, CPoint cpPoint)
{
RelayToolTipEvent(cpPoint, WM_RBUTTONDOWN);
MWContext * pMWContext = GetContext();
// Only do this if clicking is enabled, we have a context,
// or context is not destoyed
if(IsClickingEnabled() == FALSE ||
pMWContext == NULL ||
IsDestroyed()) {
return;
}
XY xyPoint;
ResolvePoint(xyPoint, cpPoint);
#ifdef LAYERS
if (pMWContext->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
event.type = CL_EVENT_MOUSE_BUTTON_DOWN;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.x = xyPoint.x;
event.y = xyPoint.y;
event.modifiers = (uFlags & MK_SHIFT ? EVENT_SHIFT_MASK : 0)
| (uFlags & MK_CONTROL ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.which = 3;
CL_DispatchEvent(pMWContext->compositor, &event);
}
else
OnRButtonDownForLayerCX(uFlags, cpPoint, xyPoint, NULL);
// Have the mouse timer handler do some dirty work.
MouseTimerData mt(pMWContext);
FEU_MouseTimer(&mt);
return;
}
BOOL
CWinCX::OnRButtonDownForLayerCX(UINT uFlags, CPoint& cpPoint,
XY& Point, CL_Layer *layer)
{
// With LAYERS turned on, the orginal method
// OnRButtonDownCX is separated into two methods,
// one of which is a per-layer method.
#endif /* LAYERS */
// Remember....
m_LastMouseEvent = m_RBDown;
m_cpRBDown = cpPoint;
m_uMouseFlags = uFlags;
return FALSE;
}
void CWinCX::OnRButtonUpCX(UINT uFlags, CPoint cpPoint)
{
RelayToolTipEvent(cpPoint, WM_RBUTTONUP);
// Only do this if clicking is enabled.
if(IsClickingEnabled() == FALSE) {
return;
}
// Don't continue if this context is destroyed.
if(IsDestroyed()) {
return;
}
XY xyPoint;
ResolvePoint(xyPoint, cpPoint);
if (GetContext()->compositor) {
CL_Event event;
fe_EventStruct fe_event;
fe_event.uFlags = uFlags;
fe_event.x = cpPoint.x;
fe_event.y = cpPoint.y;
event.type = CL_EVENT_MOUSE_BUTTON_UP;
event.fe_event = (void *)&fe_event;
event.fe_event_size = sizeof(fe_EventStruct);
event.x = xyPoint.x;
event.y = xyPoint.y;
event.modifiers = (uFlags & MK_SHIFT ? EVENT_SHIFT_MASK : 0)
| (uFlags & MK_CONTROL ? EVENT_CONTROL_MASK : 0)
| (GetKeyState(VK_MENU) < 0 ? EVENT_ALT_MASK : 0);
event.which = 3;
CL_DispatchEvent(GetContext()->compositor, &event);
}
else
OnRButtonUpForLayerCX(uFlags, cpPoint, xyPoint, NULL);
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(GetContext());
FEU_MouseTimer(&mt);
}
void
CWinCX::OnRButtonUpForLayerCX(UINT uFlags, CPoint& cpPoint,
XY& Point, CL_Layer *layer)
{
// With LAYERS turned on, the orginal method
// OnRButtonUpCX is separated into two methods,
// one of which is a per-layer method.
// Remember....
m_LastMouseEvent = m_RBUp;
m_cpLBUp = cpPoint;
m_uMouseFlags = uFlags;
}
CWnd *CWinCX::GetDialogOwner() const {
CGenericView *pView = m_pGenView;
CWnd *pOwner = NULL;
CWnd *pWnd;
if(pView) {
pOwner = pView->GetOwner();
}
if(pOwner == NULL) {
// No owner, we must take it.
CWnd * pWnd = pView->GetParentFrame();
if (pWnd)
pOwner = pWnd;
pOwner = (CWnd *)pView;
}
pWnd = pOwner->GetParentFrame();
if (pWnd)
pOwner = pWnd;
// When a modal dialog (usually a preference dialog) is active,
// it should be the message diaolg's parent,
// else user can loose alert message behind it and user can interact
// with preference dialog with bad consequences!
return(pOwner->GetLastActivePopup());
}
int CWinCX::GetUrl(URL_Struct *pUrl, FO_Present_Types iFormatOut, BOOL bReallyLoading, BOOL bForceNew)
{
// If we are POSTing data (publishing), or forcing a new page, don't ask user to save current page first
// If user cancels when being prompted to save current document, return without action
#ifdef EDITOR
if ( !bForceNew && pUrl->method != URL_POST_METHOD
&& !FE_CheckAndSaveDocument(GetContext())
) {
return( MK_NO_ACTION );
}
#endif // EDITOR
// ResetToolTipImg();
#ifdef EDITOR
#ifdef XP_WIN32
// If we are talking to LiveWire communications system,
// tell Site Manager we are about to load a new URL
// if we are NOT simply creating a new document
if ( GetContext() &&
pUrl && pUrl->address &&
0 != XP_STRCMP(pUrl->address, EDT_NEW_DOC_URL) &&
bSiteMgrIsActive ) {
pITalkSMClient->LoadingURL(pUrl->address);
}
#endif // XP_WIN32
#endif // EDITOR
// Enable the wait cursor.
SetCursor(theApp.LoadStandardCursor(IDC_WAIT));
// We need to disable/deactivate any embedded items.
OnDeactivateEmbedCX();
#ifdef EDITOR
if( EDT_IS_EDITOR(GetContext()) ){
FE_DestroyCaret(GetContext());
}
#endif
//Caret will be shown automatically after new URL is loaded
// Save the frame's URL bar text.
// We use this to determine wether or not the user has
// changed the URL bar since the new load began, if
// if so, we won't blow away what they've typed.
if(GetContext()->type == MWContextBrowser && !EDT_IS_EDITOR(GetContext())){
IChrome *pChrome = GetFrame()->GetChrome();
CWnd *pWnd = pChrome ? pChrome->GetToolbar(ID_LOCATION_TOOLBAR) : NULL;
if (pWnd && pWnd->IsKindOf(RUNTIME_CLASS(CURLBar))){
CURLBar *pUrlBar = (CURLBar *) pWnd;
if(pUrlBar != NULL)
pUrlBar->m_pBox->GetWindowText(m_csSaveLocationBarText);
else
m_csSaveLocationBarText.Empty();
}
else
m_csSaveLocationBarText.Empty();
}
else
m_csSaveLocationBarText.Empty();
// For dialog context's, we want to raise them to the top when they
// begin a load (view source, doc info, html dialogs).
if(GetContext()->type == MWContextDialog && GetFrame()->GetFrameWnd() != NULL &&
GetFrame()->GetFrameWnd()->IsWindowEnabled()) {
// Bring it to the front.
GetFrame()->GetFrameWnd()->BringWindowToTop();
// Now if it was an icon, bring it back up.
if(GetFrame()->GetFrameWnd()->IsIconic()) {
GetFrame()->GetFrameWnd()->ShowWindow(SW_RESTORE);
}
}
// Call the base.
return(CPaneCX::GetUrl(pUrl, iFormatOut, bReallyLoading, bForceNew));
}
CNSToolTip* CWinCX::CreateToolTip(LO_ImageStruct* pImage, CPoint& cpPoint, CL_Layer *layer)
{
// Added tool tip to the image.
if ((!pImage || !pImage->image_attr) ) return NULL; // image is not ready yet.
// Layout wants this in pixels, not FE units.
LTRB Rect;
ResolveElement(Rect, IL_GetImagePixmap(pImage->image_req), pImage->x_offset, pImage->y_offset,
pImage->x, pImage->y, pImage->width, pImage->height);
CPoint cpMap((int) (cpPoint.x - Twips2PixX(Rect.left)),
(int) (cpPoint.y - Twips2PixY(Rect.top)));
char* alt_text;
lo_MapRec *map;
lo_MapAreaRec_struct* loMapRec;
LO_AnchorData *pAnchorData = LO_MapXYToAreaAnchor(GetDocumentContext(), pImage,
cpMap.x, cpMap.y);
if(pImage == pLastToolTipImg && m_pLastToolTipAnchor == pAnchorData)
{
return NULL;
}
if(pAnchorData != NULL)
PA_LOCK(alt_text, char *, pAnchorData->alt);
else
PA_LOCK(alt_text, char *, pImage->alt);
pLastToolTipImg = pImage;
m_pLastToolTipAnchor = pAnchorData;
delete m_ToolTip;
if (alt_text && (*alt_text)) {
m_ToolTip = new CNSToolTip();
m_ToolTip->Create(CWnd::FromHandle(GetPane()), TTS_ALWAYSTIP);
if (::IsWindow(m_ToolTip->GetSafeHwnd())){
m_ToolTip->SetCSID( INTL_GetCSIWinCSID(LO_GetDocumentCharacterSetInfo( GetDocumentContext() )));
CRect rect(0, 0, 30, 10);
m_ToolTip->AddTool(CWnd::FromHandle(GetPane()), "", &rect, 1);
m_ToolTip->Activate(TRUE);
m_ToolTip->SetDelayTime(250);
if(pAnchorData != NULL) {
map = (lo_MapRec*)pImage->image_attr->usemap_ptr;
loMapRec = map->areas;
BOOL done = FALSE;
int tempPoint[4];
while (loMapRec != map->areas_last) {
if (loMapRec->anchor == pAnchorData && loMapRec->coords) {
tempPoint[0] = *(loMapRec->coords);
tempPoint[1] = *(loMapRec->coords+1);
tempPoint[2] = *(loMapRec->coords+2);
tempPoint[3] = *(loMapRec->coords+3);
m_ToolTip->SetBounding(tempPoint, loMapRec->coord_cnt, pImage->x, pImage->y);
break;
}
loMapRec = loMapRec->next;
}
if ((loMapRec == map->areas_last) && (loMapRec->anchor == pAnchorData) &&
loMapRec->coords) {
tempPoint[0] = *(loMapRec->coords);
tempPoint[1] = *(loMapRec->coords+1);
tempPoint[2] = *(loMapRec->coords+2);
tempPoint[3] = *(loMapRec->coords+3);
m_ToolTip->SetBounding(tempPoint, loMapRec->coord_cnt, pImage->x, pImage->y);
}
}
else {
RECT rect;
::SetRect(&rect, CASTINT(pImage->x - GetOriginX()),
CASTINT(pImage->y - GetOriginY()),
CASTINT(pImage->x - GetOriginX() + pImage->width),
CASTINT(pImage->y - GetOriginY() + pImage->height));
int32 x_offset = CL_GetLayerXOrigin(layer);
int32 y_offset = CL_GetLayerYOrigin(layer);
::OffsetRect(&rect, (int)x_offset, (int)y_offset);
m_ToolTip->SetBounding((int*)&rect, 4);
}
m_ToolTip->UpdateTipText(alt_text, CWnd::FromHandle(GetPane()), 1);
}
else {
delete m_ToolTip;
m_ToolTip = NULL;
}
}
else
m_ToolTip = NULL;
if(pAnchorData != NULL)
PA_UNLOCK(pAnchorData->alt);
else
PA_UNLOCK(pImage->alt);
return m_ToolTip;
}
void CWinCX::ClipChildren(CWnd *pWnd, BOOL bSet)
{
CFrameGlue *pGlue = GetFrame();
if (pGlue) {
pGlue->ClipChildren(pWnd, bSet);
}
}
#ifdef DDRAW
void CWinCX::BltToScreen(LTRB& rect, DDBLTFX* fx)
{
if (m_physicWinRect.IsEmpty())
CalcWinPos();
if (m_lpDDSPrimary) {
RECT destRect;
RECT rcRect;
destRect.left = rect.left + GetWindowsXPos();
destRect.top = rect.top + GetWindowsYPos();
destRect.right = rect.right + GetWindowsXPos();
destRect.bottom = rect.bottom + GetWindowsYPos();
if (destRect.bottom > m_physicWinRect.bottom) {
destRect.bottom = m_physicWinRect.bottom;
}
if (destRect.right > m_physicWinRect.right) {
destRect.right = m_physicWinRect.right;
}
if (destRect.top > destRect.bottom) { // for scrolling case
destRect.top = destRect.top % (destRect.bottom - destRect.top);
}
if (destRect.left > destRect.right) { // for scrolling case
destRect.left = destRect.left % (destRect.right - destRect.left);
}
rcRect.left = rect.left;
rcRect.top = rect.top;
rcRect.right = rect.right;
rcRect.bottom = rect.bottom;
HRESULT err;
err = m_lpDDSBack->ReleaseDC(m_offScreenDC);
err = m_lpDDSPrimary->Blt(&destRect, m_lpDDSBack, &rcRect, DDBLT_WAIT, NULL);
if (err == DDERR_SURFACELOST) {
RestoreAllDrawSurface();
err = m_lpDDSPrimary->Blt(&destRect, m_lpDDSBack, &rcRect, DDBLT_WAIT, NULL);
}
#ifdef DEBUG_mhwang
if ( err != DD_OK) {
TRACE("CWinCX::BlttoScreen err = %x\n",
err);
}
#endif
m_lpDDSBack->GetDC(&m_offScreenDC);
}
}
#endif
#ifdef DDRAW
int CWinCX::DisplayPixmap(NI_Pixmap* image, NI_Pixmap* mask, int32 x, int32 y, int32 x_offset, int32 y_offset, int32 width, int32 height, LTRB& rect)
{
SetClipOnDrawSurface(m_lpDDSBack, (HRGN)m_pDrawable->GetClip());
CDCCX::DisplayPixmap(image, mask, x, y, x_offset, y_offset, width, height, rect);
BOOL offScreenDrawing = m_pDrawable && (m_pDrawable == m_pOffscreenDrawable);
if (m_lpDDSPrimary && !offScreenDrawing && !m_ScrollWindow) {
BltToScreen(rect, NULL);
}
return (1);
}
#endif
////////////////////////////////////////////////////////////////////////////////
// Embedded Stuff
////////////////////////////////////////////////////////////////////////////////
/*
* Front-end callback from lib/plugin that creates the window for a
* brand new plugin
*/
void CWinCX::CreateEmbedWindow(MWContext *pContext, NPEmbeddedApp *pApp)
{
if (XP_FAIL_ASSERT(pContext != NULL && pApp != NULL && pApp->np_data != NULL))
return;
LO_EmbedStruct *pEmbed = ((np_data *) pApp->np_data)->lo_struct;
if (XP_FAIL_ASSERT(pEmbed != NULL))
return;
// Register the window class
HINSTANCE hinst = AfxGetInstanceHandle();
char szClassName[] = "aPluginWinClass";
WNDCLASS wc;
if(! GetClassInfo(hinst, szClassName, &wc)) {
wc.style = 0;
wc.lpfnWndProc = DefWindowProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hinst;
wc.hIcon = NULL;
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
wc.hbrBackground = (HBRUSH) (COLOR_WINDOW + 1);
wc.lpszMenuName = (LPCSTR) NULL;
wc.lpszClassName = szClassName;
if (XP_FAIL_ASSERT(RegisterClass(&wc) != 0))
return;
}
// get the current view for param 5 of CreateWindow() below
HWND cView = PANECX(pContext)->GetPane();
// Determine location of plugin rect in PIXELS
RECT rect;
// XXX This is a hack to get stuff working. The iLocation
// parameter is passed to FE_GetEmbedSize(); unfortunately, I
// don't yet pass it down through to NPL_CreatePlugin(). As far as
// I can tell, it's always set to FE_VIEW, so I'll just hack it
// for now.
//
//GetPluginRect(pContext, pEmbed, rect, iLocation, TRUE);
GetPluginRect(pContext, pEmbed, rect, FE_VIEW, TRUE);
HWND hWnd = ::CreateWindow(szClassName,
"a Plugin Window",
WS_CHILD | (pEmbed->ele_attrmask & LO_ELE_INVISIBLE ? 0 : WS_VISIBLE),
rect.left,
rect.top,
rect.right - rect.left,
rect.bottom - rect.top,
cView,
NULL,
hinst,
NULL);
if (XP_FAIL_ASSERT(hWnd != NULL))
return;
// Allocate a new NPWindow structure to hold the plugin's window
// information.
NPWindow* pAppWin = XP_NEW(NPWindow);
if (XP_FAIL_ASSERT(pAppWin != NULL)) {
::DestroyWindow(hWnd);
return;
}
pAppWin->window = (void*)hWnd;
// set the NPWindow rect
pAppWin->x = rect.left;
pAppWin->y = rect.top;
pAppWin->width = rect.right - rect.left;
pAppWin->height = rect.bottom - rect.top;
pAppWin->type = NPWindowTypeWindow;
// Adobe hack. Turn on clip children
CFrameGlue *pGlue = WINCX(pContext)->GetFrame();
if (pGlue)
pGlue->ClipChildren(CWnd::FromHandle((HWND)pAppWin->window), TRUE);
pApp->wdata = pAppWin;
}
/*
* Front-end callback from lib/plugin that saves a plug-in's window in
* a safe place to be restored (or destroyed) later.
*/
void CWinCX::SaveEmbedWindow(MWContext *pContext, NPEmbeddedApp *pApp)
{
if (XP_FAIL_ASSERT(pContext != NULL && pApp != NULL))
return;
NPWindow* pAppWin = pApp->wdata;
if (XP_FAIL_ASSERT(pAppWin != NULL))
return;
// If this isn't a windowed plugin, we've got nothing to do...
if (pAppWin->type != NPWindowTypeWindow)
return;
// Find the first non-grid parent of the applet window
MWContext* pSafeContext = XP_GetNonGridContext(pContext);
if (XP_FAIL_ASSERT(pSafeContext != NULL))
return;
HWND parent = PANECX(pSafeContext)->GetPane();
if (XP_FAIL_ASSERT(parent != NULL))
return;
// Hide and re-parent the window. We'll restore it in GetEmbedSize()
::ShowWindow((HWND)pAppWin->window, SW_HIDE);
::SetParent((HWND)pAppWin->window, parent);
}
/*
* Front-end callback from lib/plugin that tells us to restore a
* previously saved window and reparent it to the current context.
*/
void CWinCX::RestoreEmbedWindow(MWContext *pContext, NPEmbeddedApp *pApp)
{
if (XP_FAIL_ASSERT(pContext != NULL && pApp != NULL))
return;
NPWindow* pAppWin = pApp->wdata;
if (XP_FAIL_ASSERT(pAppWin != NULL))
return;
HWND parent = WINCX(pContext)->GetPane();
if (XP_FAIL_ASSERT(parent != NULL))
return;
::SetParent((HWND)pAppWin->window, parent);
::ShowWindow((HWND)pAppWin->window, SW_SHOW);
}
/*
* Front-end callback from lib/plugin that tells us to destroy
* an embedded window
*/
void CWinCX::DestroyEmbedWindow(MWContext *pContext, NPEmbeddedApp *pApp)
{
if (XP_FAIL_ASSERT(pContext != NULL && pApp != NULL))
return;
NPWindow* pAppWin = pApp->wdata;
if (XP_FAIL_ASSERT(pAppWin != NULL))
return;
if((pAppWin->window != NULL) && (pAppWin->type == NPWindowTypeWindow)) {
// the Shockwave and WebFX plugins have bugs that set
// WS_CLIPCHILDREN without clearing it so do it for them
HWND hWndParent = ::GetParent((HWND)(DWORD)pAppWin->window);
::SetWindowLong(hWndParent, GWL_STYLE,
::GetWindowLong(hWndParent, GWL_STYLE) & ~WS_CLIPCHILDREN);
// Clear the WS_CLIPCHILDREN style for all parent windows. This
// was set to satify the Adobe wankers who create a child window that's
// in a separate process
if (! IsPrintContext()) {
CWinCX *pGCX = WINCX(pContext);
if(pGCX) {
CFrameGlue *pGlue = pGCX->GetFrame();
if(pGlue) {
pGlue->ClipChildren(CWnd::FromHandle((HWND)pAppWin->window), FALSE);
}
}
}
// destroy the plugin client area
DestroyWindow((HWND)(DWORD)pAppWin->window);
pAppWin->window = NULL;
}
XP_FREE(pAppWin);
// turn scrollbars back on
if(pApp->pagePluginType == NP_FullPage)
FE_ShowScrollBars(pContext, TRUE);
}
////////////////////////////////////////////////////////////////////////////////
// Java Stuff
////////////////////////////////////////////////////////////////////////////////
#ifdef JAVA
void PR_CALLBACK
FE_DisplayNoJavaIcon(MWContext *pContext, LO_JavaAppStruct *java_struct)
{
/* write me */
}
void* PR_CALLBACK
FE_CreateJavaWindow(MWContext *pContext, LO_JavaAppStruct *java_struct,
int32 xp, int32 yp, int32 xs, int32 ys)
{
CWinCX* ctxt = WINCX(pContext);
LJAppletData* ad = (LJAppletData*)java_struct->session_data;
HWND parent;
PR_ASSERT(ad);
// get the current view
parent = ctxt->GetPane();
/* Adjust xp and yp for their offsets within the window */
xp -= ctxt->m_lOrgX;
yp -= ctxt->m_lOrgY;
// Register the window class
HINSTANCE hinst = AfxGetInstanceHandle();
HWND hWnd = NULL;
char szClassName[] = "aJavaAppletWinClass";
WNDCLASS wc;
BOOL result = FALSE;
if((result = GetClassInfo(hinst, szClassName, &wc)) == FALSE)
{
wc.style = 0;
wc.lpfnWndProc = DefWindowProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hinst;
wc.hIcon = NULL;
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
wc.hbrBackground = (HBRUSH) (COLOR_WINDOW + 1);
wc.lpszMenuName = (LPCSTR) NULL;
wc.lpszClassName = szClassName;
result = RegisterClass(&wc);
}
if(result != FALSE) {
hWnd = ::CreateWindow(szClassName,
"a JavaApplet Window",
WS_CHILD | WS_CLIPCHILDREN,
xp,
yp,
xs,
ys,
parent,
NULL,
hinst,
NULL);
}
/* set the default naviagtor palette for use by awt */
ad->fe_data = (void*)ctxt->GetPalette();
return (void*)hWnd;
}
void* PR_CALLBACK
FE_GetAwtWindow(MWContext *pContext, LJAppletData* ad)
{
if (::GetWindow((HWND)ad->window,GW_CHILD)==NULL) {
return NULL;
}
/* Above was added as an interim fix just so we can release 4.02.
The problem was that GetWindow is returning a near pointer and
this routine is simply putting DS into DX so it can return a far
pointer to its caller. But if GetWindow returns null (AX=0), it
is still putting DS into DX so it returns DS:0 to it s caller
rather than 0:0. Probably a casting bug but it needs further
investigation before a final fix is made -- Steve Morse */
return (void*)::GetWindow((HWND)ad->window, GW_CHILD);
}
void PR_CALLBACK
FE_RestoreJavaWindow(MWContext *pContext, LJAppletData* ad,
int32 xp, int32 yp, int32 xs, int32 ys)
{
CWinCX* ctxt = WINCX(pContext);
/* Adjust xp and yp for their offsets within the window */
xp -= ctxt->m_lOrgX;
yp -= ctxt->m_lOrgY;
XP_TRACE(("Restore: win=%x isWin=%d\n", ad->window, ::IsWindow((HWND)ad->window)));
if (pContext->is_grid_cell) {
/*
** Reparent the window onto the current window in case it got
** moved from a grid cell when stopped.
*/
HWND parent = ctxt->GetPane();
PR_ASSERT( ::IsWindow( (HWND)ad->window) );
::SetParent( (HWND)ad->window, parent );
}
/*
** Set the CLIPCHILDREN style for the parent window to prevent
** EraseBackground events from obliterating the java window
*/
ctxt->ClipChildren(CWnd::FromHandle((HWND)ad->window), TRUE);
::SetWindowPos((HWND)ad->window, NULL, xp, yp, xs, ys, SWP_NOZORDER);
}
void PR_CALLBACK
FE_SetJavaWindowPos(MWContext *pContext, void* window,
int32 xp, int32 yp, int32 xs, int32 ys)
{
CWinCX* ctxt = WINCX(pContext);
/* Adjust xp and yp for their offsets within the window */
xp -= ctxt->m_lOrgX;
yp -= ctxt->m_lOrgY;
::SetWindowPos((HWND)window, NULL, xp, yp, xs, ys, SWP_NOZORDER);
}
void PR_CALLBACK
FE_SetJavaWindowVisibility(MWContext *context, void* window, PRBool visible)
{
if (visible && !::IsWindowVisible((HWND)window))
::ShowWindow((HWND)window, SW_SHOW);
else if (!visible && ::IsWindowVisible((HWND)window))
::ShowWindow((HWND)window, SW_HIDE);
}
void PR_CALLBACK
FE_SaveJavaWindow(MWContext *pContext, LJAppletData* ad, void* pWindow)
{
CWinCX* ctxt = WINCX(pContext);
HWND window = (HWND)pWindow;
if (window == NULL || 0 == ::IsWindow(window)) return;
/* Hide the java applet window */
::ShowWindow( window, SW_HIDE );
/*
** Clear the WS_CLIPCHILDREN style for all parent windows. This
** was set to prevent EraseBackground events from obliterating
** the java window...
*/
ctxt->ClipChildren(CWnd::FromHandle(window), FALSE);
if (pContext->is_grid_cell) {
/*
** Reparent the applet window to the first non-grid window
** if it was on a grid cell because the grid will be
** destroyed.
**
** Note: The ad->context contains the non-grid context at this point.
*/
PR_ASSERT(!ad->context->is_grid_cell);
HWND parent = PANECX(ad->context)->GetPane();
::SetParent( window, parent );
}
/* Move the java applet window off screen - for now */
::SetWindowPos(window, NULL, -100, -100, 10, 10,
SWP_HIDEWINDOW | SWP_NOACTIVATE | SWP_NOREDRAW | SWP_NOZORDER);
}
void PR_CALLBACK
FE_FreeJavaWindow(MWContext *context, struct LJAppletData *appletData,
void* window)
{
HWND hwndFrame = (HWND)(DWORD)window;
if( hwndFrame ) {
DestroyWindow(hwndFrame);
}
}
#endif /* JAVA */
void CWinCX::DisplayJavaApp(MWContext *pContext, int iLocation, LO_JavaAppStruct *java_struct)
{
#ifdef JAVA
LJ_DisplayJavaApp(pContext, java_struct,
FE_DisplayNoJavaIcon,
FE_GetFullWindowSize,
FE_CreateJavaWindow,
FE_GetAwtWindow,
FE_RestoreJavaWindow,
FE_SetJavaWindowPos,
FE_SetJavaWindowVisibility);
#endif /* JAVA */
}
void CWinCX::HideJavaAppElement(MWContext *pContext, LJAppletData * session_data)
{
#ifdef JAVA
LJ_HideJavaAppElement(pContext, session_data, FE_SaveJavaWindow);
#endif /* JAVA */
}
void CWinCX::FreeJavaAppElement(MWContext *pContext, LJAppletData *ad)
{
#ifdef JAVA
LJ_FreeJavaAppElement(pContext, ad,
FE_SaveJavaWindow,
FE_FreeJavaWindow);
#endif /* JAVA */
}
void CWinCX::GetJavaAppSize(MWContext *pContext, LO_JavaAppStruct *java_struct,
NET_ReloadMethod reloadMethod)
{
#ifdef JAVA
LJ_GetJavaAppSize(pContext, java_struct, reloadMethod);
#else
// jevering: should this be inside ifdef JAVA?
// FE_DisplayNoJavaIcon(pContext, java_struct);
java_struct->width = 1;
java_struct->height = 1;
#endif /* ! JAVA */
}
////////////////////////////////////////////////////////////////////////////////
// End of Java Stuff
////////////////////////////////////////////////////////////////////////////////
void CWinCX::HandleClippingView(MWContext *pContext, LJAppletData *appletD, int x, int y, int width, int height)
{
}
void CWinCX::LayoutNewDocument(MWContext *pContext, URL_Struct *pURL, int32 *pWidth, int32 *pHeight, int32 *pmWidth, int32 *pmHeight)
{
// Call the base.
CPaneCX::LayoutNewDocument(pContext, pURL, pWidth, pHeight, pmWidth, pmHeight);
// Set our security indicators.
if(GetFrame()->GetMainContext() != NULL) {
GetFrame()->SetSecurityStatus(XP_GetSecurityStatus(GetFrame()->GetMainContext()->GetContext()));
}
// We're beginning to layout.
m_bIsLayingOut = TRUE;
// Make sure we have our normal arrow cursor loaded (we loaded the wait
// in GetUrl).
SetCursor(theApp.LoadStandardCursor(IDC_ARROW));
// No old progress;
m_lOldPercent = 0;
// Update the Frame's URL bar,
// only if what is currently there hasn't changed, and only
// if we're not a grid cell.
CString csText;
if(GetContext()->type == MWContextBrowser && !EDT_IS_EDITOR(GetContext())) {
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
CWnd *pWnd = pChrome->GetToolbar(ID_LOCATION_TOOLBAR);
if (pWnd && pWnd->IsKindOf(RUNTIME_CLASS(CURLBar))){
CURLBar *pUrlBar = (CURLBar *) pWnd;
if(pUrlBar != NULL) {
pUrlBar->m_pBox->GetWindowText(csText);
if(m_csSaveLocationBarText == csText && IsGridCell() == FALSE) {
pUrlBar->UpdateFields(pURL->address);
}
}
else
csText.Empty();
}
else
csText.Empty();
}
}
else
csText.Empty();
// Clear those old elements that we tracked in the previously
// loaded page (see FireMouseOverEvent....)
m_pLastOverAnchorData = NULL;
m_pLastOverElement = NULL;
m_pStartSelectionCell = NULL;
m_bLastOverTextSet = FALSE;
m_pLastImageObject = NULL;
//#ifndef NO_TAB_NAVIGATION
m_lastTabFocus.pElement = NULL;
m_lastTabFocus.mapAreaIndex = 0; // 0 means no focus, start with index 1.
m_lastTabFocus.pAnchor = NULL;
m_isReEntry_setLastTabFocusElement = 0; // to provent re-entry
SetMainFrmTabFocusFlag(CMainFrame::TAB_FOCUS_IN_NULL); // I don't have focus.
//#endif /* NO_TAB_NAVIGATION */
::SetTextAlign(GetContextDC(),TA_NOUPDATECP);
} // LayoutNewDocument()
void CWinCX::SetMainFrmTabFocusFlag( int nn )
{
CFrameGlue * pFrame = GetFrame();
if( pFrame ) {
CFrameWnd * pFrameWindow = pFrame->GetFrameWnd();
if( pFrameWindow && pFrameWindow->IsKindOf(RUNTIME_CLASS(CMainFrame))) {
((CMainFrame *)pFrameWindow)->SetTabFocusFlag( nn );
}
}
}
void CWinCX::FinishedLayout(MWContext *pContext) {
// Call the base.
CPaneCX::FinishedLayout(pContext);
// We're no longer laying out.
m_bIsLayingOut = FALSE;
// Progress should be maxed.
m_lOldPercent = 100;
}
void CWinCX::AllConnectionsComplete(MWContext *pContext)
{
// Call the base.
CDCCX::AllConnectionsComplete(pContext);
// Stop our frame's animation, if the main context of the frame is no longer busy.
if(GetFrame()->GetMainContext()) {
if(XP_IsContextBusy(GetFrame()->GetMainContext()->GetContext()) == FALSE) {
// Okay, stop the animation.
StopAnimation();
// Also, we can clear the progress bar now.
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
LPNSSTATUSBAR pIStatusBar = NULL;
pChrome->QueryInterface( IID_INSStatusBar, (LPVOID *) &pIStatusBar );
if ( pIStatusBar ) {
pIStatusBar->SetProgress(0);
pIStatusBar->Release();
}
}
// We need to make sure the toolbar buttons are correctly updated. If we
// don't force it now it won't happen until the app goes idle (which could
// be when the user moves the mouse over the window, for example)
CGenericView *pView = GetView();
ASSERT(pView);
CFrameWnd* pFrameWnd = pView->GetParentFrame();
ASSERT(pFrameWnd);
if (pFrameWnd) {
pFrameWnd->SendMessageToDescendants(WM_IDLEUPDATECMDUI, (WPARAM)TRUE, (LPARAM)0);
}
}
}
if( theGlobalNSFont.WebfontsNeedReload( pContext ) )
{
// need to remove all font cache before reload.
ClearFontCache();
int usePassInType = 1;
NiceReload(usePassInType, NET_RESIZE_RELOAD );
}
}
void CWinCX::UpdateStopState(MWContext *pContext)
{
#ifdef XP_WIN32
// Force the toolbar buttons to be correctly updated. If we
// don't force it now it won't happen until the app goes idle (which could
// be when the user moves the mouse over the window, for example)
CGenericView *pView = GetView();
if(pView) {
CFrameGlue *pGlue = GetFrame();
CFrameWnd *pFWnd = pGlue->GetFrameWnd();
if(pFWnd) {
pFWnd->SendMessageToDescendants(WM_IDLEUPDATECMDUI, (WPARAM)TRUE, (LPARAM)0);
}
}
#endif
}
// Note: We now use pTitle = NULL to clear an existing title
void CWinCX::SetDocTitle(MWContext *pContext, char *pTitle)
{
// Call the base.
CDCCX::SetDocTitle(pContext, pTitle);
MWContext * pMWContext = GetContext();
BOOL bIsPageComposer = EDT_IS_EDITOR(pMWContext) && (pMWContext->type != MWContextMessageComposition);
// Guard against the case where our window has gone away and the
// closing of the window causes the stream to complete which
// causes us to get back in here with a half torn down window
if( GetDocumentContext() == NULL )
return;
// We need to be the main context in order to set a
// frames title.
if( GetFrame()->GetMainContext() == this ) {
// Munge the title string.
CString csTitle;
CString csUrlTitle = pTitle;
// This should be set at end of GetUrl so we
// don't have to depend on history
CString csBaseURL = LO_GetBaseURL( GetDocumentContext() );
BOOL bTitleIsSameAsUrl = (csBaseURL == csUrlTitle);
if(!csUrlTitle.IsEmpty()) {
if ( bTitleIsSameAsUrl ) {
// We won't be adding on the URL after this,
// so make a bigger title
// If we are a URL, cut from the middle
WFE_CondenseURL(csUrlTitle, 50, FALSE);
} else {
if ( bIsPageComposer ) {
// Use just left portion of title
csUrlTitle = csUrlTitle.Left(30);
} else {
// We won't add URL to Browser, so use more
csUrlTitle = csUrlTitle.Left(100);
}
}
csTitle += csUrlTitle;
}
if( bIsPageComposer && !bTitleIsSameAsUrl ){
// Append URL if we didn't already use it as the title
// Limit text inside of ( )
if( !csUrlTitle.IsEmpty() ){
csTitle += " : ";
}
// Strip off username and password from URL
char * pStripped = NULL;
NET_ParseUploadURL( (char*)LPCSTR(csBaseURL), &pStripped, NULL, NULL );
csBaseURL = pStripped;
XP_FREEIF(pStripped);
WFE_CondenseURL(csBaseURL, 50 - (min(csUrlTitle.GetLength(),20)), FALSE);
csTitle += csBaseURL;
}
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
pChrome->SetDocumentTitle(csTitle);
}
}
}
void CWinCX::ClearView(MWContext *pContext, int iView) {
// Call the base.
CDCCX::ClearView(pContext, iView);
// Have the view erase it's background to clear.
RECT crClear;
::SetRect(&crClear, 0, 0, (int) GetWidth(), (int) GetHeight());
if(GetPane()) {
// Must first update the window, to get rid of any
// queud erase backgrounds and stuff which will
// cause the display to become corrupted.
RECT tempRect;
::SetRect(&tempRect,0, 0, CASTINT(GetWidth()), CASTINT(GetHeight()));
::InvalidateRect(GetPane(), &tempRect, FALSE);
// ::UpdateWindow(GetPane());
}
else {
// Tell layout to specifically refresh our area.
RefreshArea(GetOriginX(), GetOriginY(), GetWidth(), GetHeight());
}
}
void CWinCX::SetDocDimension(MWContext *pContext, int iLocation, int32 lWidth, int32 lLength) {
// This saves old Y, which changes under certain delete situations...
int32 iOriginY = GetOriginY();
// Make sure that the origin is still visible. (This case only matters when the document
// height shrinks, which currently only happens when editing.)
m_lOrgY = max(0, min(m_lOrgY, lLength - m_lHeight));
// Anytime we change the scrolling origin, we have to tell the compositor.
if ( GetContext()->compositor)
CL_ScrollCompositorWindow(GetContext()->compositor, m_lOrgX, m_lOrgY);
// Call the base.
CPaneCX::SetDocDimension(pContext, iLocation, lWidth, lLength);
#ifdef EDITOR
if(EDT_IS_EDITOR(pContext) &&
GetOriginY() != iOriginY ) {
// Redraw entire view
// This is needed instead to be sure caret is moved to correct location
// after doc size has changed, like after a deletion
// at bottom of doc (bug 65199)
EDT_RefreshLayout(pContext);
}
#endif /* EDITOR */
}
//
// Fake scroll messages so that we use scrollwindowex() to move the window
// so that our form elements actually move
//
void CWinCX::SetDocPosition(MWContext *pContext, int iLocation, int32 lX, int32 lY)
{
int iPos;
// If we're (the window) wider than the document (layout),
// then don't use the X value.
if(GetDocumentWidth() <= GetWidth()) {
lX = GetOriginX();
}
int32 lRemY = GetOriginY();
int32 lRemX = GetOriginX();
// Call the base.
CDCCX::SetDocPosition(pContext, iLocation, lX, lY);
// LTNOTE: in the editor documents can shrink. Editor contexts therefor
// must always scroll.
// Scrolling in On?ScrollCX is lossy, and causes wild results
// (scrolling right will cause to scroll right and up).
// Make sure there is a need to scroll before attempting to scroll.
// scroll to the correct Y location
if((m_lDocHeight - m_lHeight > 0) && lRemY != lY) {
iPos = (int) ((double) lY * (double) GetPageY() / (double) (m_lDocHeight - m_lHeight));
Scroll(SB_VERT, SB_THUMBTRACK, iPos, NULL);
}
else if ( EDT_IS_EDITOR( pContext ) ){
if((m_lDocHeight - m_lHeight > 0)) {
iPos = (int) ((double) lY * (double) GetPageY() / (double) (m_lDocHeight - m_lHeight));
Scroll(SB_VERT, SB_THUMBTRACK, iPos, NULL);
}
else {
Scroll(SB_VERT, SB_THUMBTRACK, 0, NULL);
}
}
// now do X
if((m_lDocWidth - m_lWidth > 0) && lRemX != lX) {
iPos = (int) ((double) lX * (double) GetPageX() / (double) (m_lDocWidth - m_lWidth));
Scroll(SB_HORZ, SB_THUMBTRACK, iPos, NULL);
}
else if( EDT_IS_EDITOR(pContext) ){
if((m_lDocWidth - m_lWidth > 0)) {
iPos = (int) ((double) lX * (double) GetPageX() / (double) (m_lDocWidth - m_lWidth));
Scroll(SB_HORZ, SB_THUMBTRACK, iPos, NULL);
}
else {
Scroll(SB_HORZ, SB_THUMBTRACK, 0, NULL);
}
}
// realize scrollbars will have been called by the scrolling routines
}
//#ifndef NO_TAB_NAVIGATION
// CWinCX::DisplayFeedback() is called by CFE_DisplayFeedback() to draw image selection feedback,
// and tab focus.
//
// For clear calling relation, CDCCX::DisplayFeedback() is renamed to CDCCX::DisplayImageFeedback()
// and added theArea, drawFlag parameters.
void CWinCX::DisplayFeedback(MWContext *pContext, int iLocation, LO_Element *pElement)
{
if( pElement->lo_any.type == LO_IMAGE ) {
// Only if the pImage has the tab focus, can we get the tab focus area.
// otherwise it will return NULL.
lo_MapAreaRec * theArea = NULL;
uint32 drawFlag = 0;
getImageDrawFlag(pContext, (LO_ImageStruct *)pElement, &theArea, &drawFlag);
CDCCX::DisplayImageFeedback(pContext, iLocation, pElement, theArea, drawFlag );
}
// do nothing if it is not image.
}
//#endif /* NO_TAB_NAVIGATION */
void CWinCX::SetProgressBarPercent(MWContext *pContext, int32 lPercent) {
// Enusre the safety of the value.
lPercent = lPercent < -1 ? 0 : ( lPercent > 100 ? 100 : lPercent );
if ( m_lOldPercent == lPercent ) {
return;
}
m_lOldPercent = lPercent;
if (GetFrame()) {
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
LPNSSTATUSBAR pIStatusBar = NULL;
pChrome->QueryInterface( IID_INSStatusBar, (LPVOID *) &pIStatusBar );
if ( pIStatusBar ) {
pIStatusBar->SetProgress(CASTINT(lPercent));
pIStatusBar->Release();
}
}
}
}
void CWinCX::Progress(MWContext *pContext, const char *pMessage) {
if (GetFrame()) {
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
LPNSSTATUSBAR pIStatusBar = NULL;
pChrome->QueryInterface( IID_INSStatusBar, (LPVOID *) &pIStatusBar );
if ( pIStatusBar ) {
pIStatusBar->SetStatusText(pMessage);
pIStatusBar->Release();
}
}
}
}
void CWinCX::DisplayEdge(MWContext *pContext, int iLocation, LO_EdgeStruct *pEdge) {
// Create an edge window if none currently exists.
if(pEdge->FE_Data == NULL) {
CGridEdge *pNewEdge = new CGridEdge(pEdge, this);
pEdge->FE_Data = pNewEdge;
}
else {
// Update the object's idea of it's owner in case layout is swapping objects on us,
// this in turn causes the edge to display in a possibly new location.
((CGridEdge *)pEdge->FE_Data)->UpdateEdge(pEdge);
}
}
void CWinCX::FreeEdgeElement(MWContext *pContext, LO_EdgeStruct *pEdge) {
// Get rid of the Edge.
if(pEdge && pEdge->FE_Data) {
delete((CGridEdge *)(pEdge->FE_Data));
pEdge->FE_Data = NULL;
}
}
CWinCX *CWinCX::DetermineTarget(const char *pTargetName) {
// This function decides what target will load a URL.
// This is to facilitate grids loading into targeted locations.
// Ensure the target name exists.
if(pTargetName == NULL || *pTargetName == '\0') {
// If we're a dialog, and we can not close ourselves, and we're modal over another
// window context, send the url to that context instead.
if(GetContext()->type == MWContextDialog &&
m_cplModalOver.IsEmpty() == FALSE &&
GetDocument() &&
GetDocument()->CanClose() == FALSE) {
HWND hDestination = (HWND)m_cplModalOver.GetHead();
if(IsWindow(hDestination)) {
CWnd *pDestination = CWnd::FromHandlePermanent(hDestination);
if(pDestination && pDestination->IsKindOf(RUNTIME_CLASS(CGenericFrame))) {
CGenericFrame *pFrame = (CGenericFrame *)pDestination;
if(pFrame->GetActiveContext() &&
pFrame->GetActiveContext()->IsWindowContext() &&
pFrame->GetActiveContext()->IsDestroyed() == FALSE) {
// GARRETT: This needs to support PANECX
return(VOID2CX(pFrame->GetActiveContext(), CWinCX)->DetermineTarget(pTargetName));
}
}
}
}
return(this);
}
// Attempt to find the context already in existance.
MWContext *pCandidate = XP_FindNamedContextInList(GetContext(), (char *)pTargetName);
if(pCandidate != NULL) {
if(ABSTRACTCX(pCandidate)->IsWindowContext() == FALSE) {
// Don't allow anything but windows to take this burden.
return(this);
}
else {
return(VOID2CX(pCandidate->fe.cx, CWinCX));
}
}
// Create a new one, dammit.
pCandidate = FE_MakeBlankWindow(GetContext(), NULL, (char *)pTargetName);
if(pCandidate == NULL) {
// Just use ourselves, I see no way out.
return(this);
}
return(VOID2CX(pCandidate->fe.cx, CWinCX));
}
int32 CWinCX::QueryProgressPercent() {
// If we've no children, return our percentage.
if(IsGridParent() == FALSE) {
return(m_lOldPercent);
}
// It's not empty.
// We need to return the sum of each of our grid children, divided by the number
// of the grid children.
int32 lCount = XP_ListCount(GetContext()->grid_children);
int32 lSum = 0;
// Go through each child.
MWContext *pChild;
XP_List *pTraverse = GetContext()->grid_children;
while (pChild = (MWContext *)XP_ListNextObject(pTraverse)) {
ASSERT(ABSTRACTCX(pChild)->IsWindowContext());
lSum += WINCX(pChild)->QueryProgressPercent();
}
// just in case...
if (lCount < 1)
return(m_lOldPercent);
// Return the sum divided by the count.
return(lSum / lCount);
}
void CWinCX::StopAnimation()
{
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
pChrome->StopAnimation();
}
}
void CWinCX::StartAnimation()
{
LPCHROME pChrome = GetFrame()->GetChrome();
if(pChrome) {
pChrome->StartAnimation();
}
}
void CWinCX::OpenFile() {
// If we've a parent context, let them deal with this.
if(GetParentContext()) {
if(ABSTRACTCX(GetParentContext())->IsWindowContext()) {
WINCX(GetParentContext())->OpenFile();
}
else {
ASSERT(0);
}
}
else if(IsDestroyed() == FALSE) {
// Have the frame bring up the open file dialog.
if(GetFrame()->GetFrameWnd()) {
MWContext * pMWContext = GetContext();
// Restrict opening to only HTML files if initiated from Composer frame
BOOL bOpenIntoEditor = EDT_IS_EDITOR(pMWContext) && (pMWContext->type == MWContextBrowser);
CWnd * cWnd = GetFrame()->GetFrameWnd();
char * pName = wfe_GetExistingFileName(
cWnd->m_hWnd,
szLoadString(IDS_OPEN), bOpenIntoEditor ? HTM_ONLY : HTM, TRUE);
if(pName == NULL) {
// Canceled.
return;
}
#ifdef EDITOR
if( bOpenIntoEditor ){
FE_LoadUrl(pName, LOAD_URL_COMPOSER);
} else {
if( EDT_IS_EDITOR(pMWContext) ){
// Open a file into a new browse window from an Editor
FE_LoadUrl(pName, LOAD_URL_NAVIGATOR);
} else {
// Load URL into existing Browser window
NormalGetUrl(pName);
}
}
#else
NormalGetUrl(pName);
#endif // EDITOR
XP_FREE(pName);
}
else {
ASSERT(0);
}
}
}
BOOL CWinCX::CanOpenFile() {
// If we've a parent context, let them deal with this.
if(GetParentContext()) {
if(ABSTRACTCX(GetParentContext())->IsWindowContext()) {
return(WINCX(GetParentContext())->CanOpenFile());
}
else {
ASSERT(0);
return(FALSE);
}
}
else if(IsDestroyed() == TRUE) {
return(FALSE);
}
else {
// Can always open a file.
// Perhaps not if we're in some kiosk mode.
return(TRUE);
}
}
BOOL CWinCX::SaveOleDocument() {
CGenericDoc * pDoc = GetDocument();
return pDoc->OnSaveDocument(NULL);
}
void CWinCX::SaveAs() {
if (!SaveOleDocument()) {
if(IsDestroyed() == FALSE && IsGridParent() == FALSE) {
// Make sure we have a history entry.
History_entry *pHist = SHIST_GetCurrent(&(GetContext()->hist));
if(pHist != NULL && pHist->address != NULL) {
// Can save.
// Don't care if GetFrameWnd returns NULL.
CSaveCX::SaveAnchorObject((const char *)pHist->address,
pHist, GetFrame()->m_iCSID, GetFrame()->GetFrameWnd());
}
}
}
}
BOOL CWinCX::CanSaveAs() {
// if there is only 1 ole item there, don't save the documant.
CGenericDoc * pDoc = GetDocument();
POSITION pos = pDoc->GetStartPosition();
CNetscapeCntrItem *pItem;
CDocItem* item = pDoc->GetNextItem(pos );
// If the current embed item is full page OLE and is not in-placed actived, return FALSE
// so the saveAs menu item will be disabled.
while (item && item->IsKindOf(RUNTIME_CLASS(CNetscapeCntrItem))) {
pItem = (CNetscapeCntrItem*)item;
if (pItem->m_isFullPage && !pItem->IsInPlaceActive( ))
return FALSE;
item = pDoc->GetNextItem(pos );
}
pItem = (CNetscapeCntrItem *)pDoc->GetInPlaceActiveItem(CWnd::FromHandle(GetPane()));
if (pItem) {
if (pItem->m_bCanSavedByOLE) // see can this item be saved.
return TRUE;
else
return FALSE;
}
if(IsDestroyed() == TRUE || IsGridParent() == TRUE) {
return(FALSE);
}
else {
// Can't save if there's no history entry.
History_entry *pHist = SHIST_GetCurrent(&(GetContext()->hist));
if(pHist == NULL) {
return(FALSE);
}
else if(pHist->address == NULL) {
return(FALSE);
}
else {
return(TRUE);
}
}
}
void CWinCX::PrintContext() {
//Abstraction layer has context type print which
//conflicts with abstracting the Print() function
Print();
}
void CWinCX::Print() {
if(IsDestroyed() == TRUE || CanPrint() == FALSE) {
return;
}
MWContext *pMWContext = GetContext();
if( !pMWContext )
return;
// Note: We no longer have to force saving a page
// before printing because we use a temporary file instead
// Always pass the WYSIWYG attribute for printing URLs (javascript).
SHIST_SavedData SavedData;
URL_Struct *pUrl = CreateUrlFromHist(TRUE, &SavedData, TRUE);
char *pDisplayUrl = NULL;
#ifdef EDITOR
if( EDT_IS_EDITOR(pMWContext) )
{
// Save actual address we want to show in header or footer
// to pass on to print context
if( pUrl->address )
pDisplayUrl = XP_STRDUP(pUrl->address);
// Save current contents to a temporary file if
// we are a Mail Message, a New Document, or there are changes to current page
// This will change pUrl->address to temp filename
if( !FE_PrepareTempFileUrl(pMWContext, pUrl) )
{
XP_FREEIF(pDisplayUrl);
// Failed to save to the temp file - abort
return;
}
}
#endif //EDITOR
// Copy the necessary information into the URL's saved data so that we don't
// make a copy of the plug-ins when printing
NPL_PreparePrint(pMWContext, &pUrl->savedData);
CGenericView *pView = GetView();
if(pView) {
CPrintCX::PrintAnchorObject(pUrl, pView, &SavedData, pDisplayUrl);
}
XP_FREEIF(pDisplayUrl);
}
BOOL CWinCX::CanPrint(BOOL bPreview) {
// Can't print if we're destroyed, if there's no view, or if we've no
// history or we're a grid parent.
BOOL bRetval = TRUE;
if(FALSE == sysInfo.m_bPrinter) {
bRetval = FALSE;
}
else if(IsDestroyed()) {
bRetval = FALSE;
}
else if(GetPane() == NULL) {
bRetval = FALSE;
}
else if(CanCreateUrlFromHist() == FALSE) {
bRetval = FALSE;
}
else if(IsGridParent() == TRUE) {
bRetval = FALSE;
}
// Lastly, we only allow one print context at a time.
// There can be multiple print preview context's however.
if(bRetval && bPreview == FALSE) {
MWContext *pTraverseContext = NULL;
CAbstractCX *pTraverseCX = NULL;
XP_List *pTraverse = XP_GetGlobalContextList();
while (pTraverseContext = (MWContext *)XP_ListNextObject(pTraverse)) {
if(pTraverseContext != NULL && ABSTRACTCX(pTraverseContext) != NULL) {
pTraverseCX = ABSTRACTCX(pTraverseContext);
if(pTraverseCX->IsPrintContext() == TRUE) {
CPrintCX *pPrintCX = VOID2CX(pTraverseCX, CPrintCX);
if(pPrintCX->IsPrintPreview() == FALSE) {
// Already a print job in progress.
bRetval = FALSE;
break;
}
}
}
}
}
return(bRetval);
}
void CWinCX::AllFind(MWContext *pSearchContext) {
// If we've got a parent context, let it deal.
if(GetParentContext()) {
if(ABSTRACTCX(GetParentContext())->IsWindowContext()) {
WINCX(GetParentContext())->AllFind(pSearchContext);
}
else {
ASSERT(0);
}
}
else if(CanAllFind()) {
if(GetFrame()->GetFrameWnd()) {
CNetscapeFindReplaceDialog *dlg;
dlg = new CNetscapeFindReplaceDialog();
UINT flags = FR_DOWN | FR_NOWHOLEWORD | FR_HIDEWHOLEWORD;
BOOL bBrowser = TRUE;
#ifdef EDITOR
if( pSearchContext )
{
bBrowser = !EDT_IS_EDITOR(pSearchContext);
} else {
bBrowser = !EDT_IS_EDITOR(GetContext());
}
#endif
dlg->Create(bBrowser,
theApp.m_csFindString,
bBrowser ? NULL : (LPCTSTR)theApp.m_csReplaceString,
flags,
GetFrame()->GetFrameWnd());
// Let them know who the frame is.
dlg->SetFrameGlue(GetFrame());
// If a frame was specified as a search context then perform
// the search there. If no context was specified or if
// the search context is a top level window then don't set
// the search context and we'll search normally on the focused frame.
if (pSearchContext && pSearchContext->grid_parent)
dlg->SetSearchContext(pSearchContext);
}
else {
ASSERT(0);
}
}
}
BOOL CWinCX::CanAllFind() {
// If we've a parent context, let them deal with this.
if(GetParentContext()) {
if(ABSTRACTCX(GetParentContext())->IsWindowContext()) {
return(WINCX(GetParentContext())->CanAllFind());
}
else {
ASSERT(0);
return(FALSE);
}
}
else if(IsDestroyed() == TRUE) {
return(FALSE);
}
else {
BOOL bRetval = TRUE;
// Can't find if there's no history entry.
History_entry *pHist = SHIST_GetCurrent(&(GetContext()->hist));
if(pHist == NULL) {
bRetval = FALSE;
}
else if(pHist->address == NULL) {
bRetval = FALSE;
}
else if(GetFrame()->CanFindReplace() == FALSE) {
// Frame already is finding/replacing.
bRetval = FALSE;
}
else {
bRetval = TRUE;
}
return(bRetval);
}
}
//
// Find again using the settings from last time
//
void CWinCX::FindAgain()
{
DoFind(NULL, (const char *) theApp.m_csFindString, theApp.m_bMatchCase,
theApp.m_bSearchDown, TRUE);
}
//
// Actually do the find operation
//
BOOL CWinCX::DoFind(CWnd * pWnd, const char * pFindString, BOOL bMatchCase,
BOOL bSearchDown, BOOL bAlertOnNotFound)
{
int32 start_position, end_position;
LO_Element * start_ele_loc = NULL;
LO_Element * end_ele_loc = NULL;
CL_Layer *sel_layer = NULL;
// Start the search from the current selection location
LO_GetSelectionEndpoints(GetDocumentContext(),
&start_ele_loc,
&end_ele_loc,
&start_position,
&end_position,
&sel_layer);
// look for the text
if (LO_FindText(GetDocumentContext(),
(char *) pFindString,
&start_ele_loc,
&start_position,
&end_ele_loc,
&end_position,
bMatchCase,
bSearchDown) == 1)
{
int32 x, y;
LO_SelectText(GetDocumentContext(),
start_ele_loc,
start_position,
end_ele_loc,
end_position,
&x,
&y);
// TODO: This is not correct - it scrolls over unnecessarily
//FE_SetDocPosition(GetContext(), FE_VIEW, x, y);
// Get current location and scroll only as necessary
int32 iViewLeft, iViewTop, iViewWidth, iViewHeight;
FE_GetDocAndWindowPosition(GetContext(), &iViewLeft, &iViewTop, &iViewWidth, &iViewHeight);
int32 iViewBottom = iViewTop + iViewHeight;
int32 iViewRight = iViewLeft + iViewWidth;
BOOL bScroll = FALSE;
if( x < iViewLeft || x > iViewRight )
{
iViewLeft = max(0,(x - iViewWidth/4));
bScroll = TRUE;
}
// Place found text at 1/4 of window height
if( y < iViewTop || y > iViewBottom )
{
iViewTop = max(0,(y - iViewHeight/4));
bScroll = TRUE;
}
if( bScroll )
FE_SetDocPosition(GetContext(), FE_VIEW, iViewLeft, iViewTop);
//
// Ensure the Find/Replace dlg doesn't cover the selection
//
CNetscapeFindReplaceDialog *pDlg = GetFrame() ? GetFrame()->GetFindReplace() : NULL;
if( pDlg )
{
RECT rcDlg;
pDlg->GetWindowRect( &rcDlg );
RECT rcView;
::GetWindowRect( GetPane(), &rcView );
int32 iTextY = rcView.top + (y - GetOriginY());
if( (rcDlg.top < iTextY) && (rcDlg.bottom > iTextY) )
{
CRect rcDlgNew( rcDlg );
rcDlgNew.OffsetRect( 0, iTextY - rcDlgNew.top + 20 );
if( rcDlgNew.bottom > GetSystemMetrics( SM_CYSCREEN ) )
{
rcDlgNew.CopyRect( &rcDlg );
rcDlgNew.OffsetRect( 0, iTextY - rcDlgNew.bottom - 1 );
}
pDlg->SetWindowPos( NULL, rcDlgNew.left, rcDlgNew.top, 0, 0, SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE );
}
}
return TRUE;
}
else if(bAlertOnNotFound){
HWND hBox = NULL;
if(NULL == pWnd) {
hBox = GetPane();
}
else {
hBox = pWnd->m_hWnd;
}
if(hBox)
::MessageBox(hBox, szLoadString(IDS_FIND_NOT_FOUND), szLoadString(AFX_IDS_APP_TITLE), MB_ICONEXCLAMATION | MB_OK);
}
return FALSE;
}
//
// cause the form containing the given element to be submitted
//
void FE_SubmitInputElement(MWContext * pContext, LO_Element * pElement)
{
LO_FormSubmitData * submit;
URL_Struct * URL_s;
if(!pContext || !pElement)
return;
submit = LO_SubmitForm(ABSTRACTCX(pContext)->GetDocumentContext(), (LO_FormElementStruct *) pElement);
if(!submit)
return;
// Create the URL to load
URL_s = NET_CreateURLStruct((char *)submit->action, NET_DONT_RELOAD);
// attach form data to the URL
NET_AddLOSubmitDataToURLStruct(submit, URL_s);
// add referer field if we've got it
{
History_entry *he = SHIST_GetCurrent (&pContext->hist);
if(he && he->address)
URL_s->referer = XP_STRDUP(he->address);
else
URL_s->referer = NULL;
}
// start the spinning icon
// Load up.
ABSTRACTCX(pContext)->GetUrl(URL_s, FO_CACHE_AND_PRESENT);
LO_FreeSubmitData(submit);
}
// Say wether or not view source is allowed.
BOOL CWinCX::CanViewSource()
{
// If we've got a parent context, we need to pass it on up the chain.
if(GetParentContext() != NULL) {
return(WINCX(GetParentContext())->CanViewSource());
}
// We're the top level context, make sure we're not destroyed.
if(IsDestroyed()) {
return(FALSE);
}
// Don't let this happen while a context is loading otherwise,
// not all the document source is present, and we interrupt
// the load.
if(XP_IsContextBusy(GetContext()) == TRUE) {
return(FALSE);
}
// All that matters now is wether or not we can create a URL struct from
// our history.
return(CanCreateUrlFromHist());
}
// View the source.
void CWinCX::ViewSource()
{
// If we've a parent context, we need to pass it on up the chain.
if(GetParentContext() != NULL) {
WINCX(GetParentContext())->ViewSource();
return;
}
if(CanViewSource()) {
// Clear the state data.
URL_Struct *pUrl = CreateUrlFromHist(TRUE);
if(pUrl != NULL) {
#ifdef EDITOR
if( EDT_IS_EDITOR(GetContext()) ){
EDT_DisplaySource(GetContext()); // Make sure that it's okay to do this.
return;
}
#endif //EDITOR
// Load up.
GetUrl(pUrl, FO_CACHE_AND_VIEW_SOURCE);
}
}
}
BOOL CWinCX::CanDocumentInfo()
{
// If we've got a parent context, we need to pass it on up the chain.
if(GetParentContext() != NULL) {
return(WINCX(GetParentContext())->CanDocumentInfo());
}
// We're the top level context, make sure we're not destroyed.
if(IsDestroyed()) {
return(FALSE);
}
// Don't allow this while we're loading.
// They won't be able to view the entire document's tree.
// Also, this won't interrupt the current load.
if(XP_IsContextBusy(GetContext()) == TRUE) {
return(FALSE);
}
// All that matters now is wether or not we can create a URL struct from
// our history.
return(CanCreateUrlFromHist());
}
void CWinCX::DocumentInfo()
{
// If we've got a parent context, we need to pass it on up the chain.
if(GetParentContext() != NULL) {
WINCX(GetParentContext())->DocumentInfo();
return;
}
// Make sure it's okay to try this.
if(CanDocumentInfo()) {
// Do it.
NormalGetUrl("about:document");
}
}
BOOL CWinCX::CanFrameSource()
{
if(IsDestroyed()) {
return(FALSE);
}
if(IsGridCell() == FALSE) {
return(FALSE);
}
// Don't let this happen while a context is loading otherwise,
// not all the document source is present, and we interrupt
// the load.
if(XP_IsContextBusy(GetContext()) == TRUE) {
return(FALSE);
}
// All that matters now is wether or not we can create a URL struct from
// our history.
return(CanCreateUrlFromHist());
}
void CWinCX::FrameSource()
{
if(CanFrameSource()) {
// Clear the state data.
URL_Struct *pUrl = CreateUrlFromHist(TRUE);
if(pUrl != NULL) {
#ifdef EDITOR
if( EDT_IS_EDITOR(GetContext()) ){
EDT_DisplaySource(GetContext()); // Make sure that it's okay to do this.
return;
}
#endif
// Load up.
GetUrl(pUrl, FO_CACHE_AND_VIEW_SOURCE);
}
}
}
BOOL CWinCX::CanFrameInfo()
{
if(IsDestroyed()) {
return(FALSE);
}
if(IsGridCell() == FALSE) {
return(FALSE);
}
// Don't allow this while we're loading.
// They won't be able to view the entire document's tree.
// Also, this won't interrupt the current load.
if(XP_IsContextBusy(GetContext()) == TRUE) {
return(FALSE);
}
// All that matters now is wether or not we can create a URL struct from
// our history.
return(CanCreateUrlFromHist());
}
void CWinCX::FrameInfo()
{
// Make sure it's okay to try this.
if(CanFrameInfo()) {
// Do it.
NormalGetUrl("about:document");
}
}
BOOL CWinCX::CanGoHome()
{
// Have the parent handle.
if(GetParentContext() != NULL) {
return(WINCX(GetParentContext())->CanGoHome());
}
// We can't be destroyed.
if(IsDestroyed()) {
return(FALSE);
}
// We need to have a home page defined.
if(theApp.m_pHomePage.IsEmpty()) {
return(FALSE);
}
return(TRUE);
}
void CWinCX::GoHome()
{
// Have the parent handle.
if(GetParentContext() != NULL) {
WINCX(GetParentContext())->GoHome();
return;
}
// Make sure we can go home.
if(CanGoHome()) {
// Load it.
NormalGetUrl(theApp.m_pHomePage);
}
}
// This function get's called to size the non client area.
// On grid cells, we like to put a pixel border on the outsize so we can show activation.
void CWinCX::OnNcCalcSizeCX(BOOL bCalcValidRects, NCCALCSIZE_PARAMS* lpncsp)
{
if(IsGridCell() == TRUE &&
IsGridParent() == FALSE &&
GetPane() != NULL &&
IsIconic(GetPane()) == FALSE &&
IsDestroyed() == FALSE &&
m_bHasBorder == TRUE
) {
if(lpncsp) {
lpncsp->rgrc[0].left += 1;
lpncsp->rgrc[0].top += 1;
lpncsp->rgrc[0].right -= 1;
lpncsp->rgrc[0].bottom -= 1;
}
}
}
// This function get's called to draw the area surrounding the client area of
// the view, if anything special needs be done.
void CWinCX::OnNcPaintCX() {
if(IsGridCell() == TRUE &&
IsGridParent() == FALSE &&
GetPane() != NULL &&
IsDestroyed() == FALSE
#ifdef XP_WIN16
&& GetFrame()->GetFrameWnd() != NULL
#endif
&& m_bHasBorder == TRUE ) {
// Get a DC from our true window, we'll be using it to do the deed, as
// we're drawing outside of our client region.
HDC hDC;
#ifdef XP_WIN16
CDC *pDC = GetFrame()->GetFrameWnd()->GetDC();
hDC = pDC->GetSafeHdc();
#else
hDC = ::GetWindowDC(GetPane());
#endif
if(hDC != NULL) {
RECT crWindowRect;
::GetWindowRect(GetPane(), &crWindowRect);
RECT crBorder;
::SetRect(
&crBorder,
0,
0,
crWindowRect.right - crWindowRect.left,
crWindowRect.bottom - crWindowRect.top);
#ifdef XP_WIN16
// 16 bit working in frame coordinates.
::MapWindowPoints(
GetPane(),
GetFrame()->GetFrameWnd()->m_hWnd,
(POINT *)&crBorder,
2);
// Need to shift up and left one pixel.
crBorder.left -= 1;
crBorder.right -= 1;
crBorder.top -= 1;
crBorder.bottom -= 1;
#endif
// Figure out what colors we are drawing with.
COLORREF rgbOuter = m_rgbBackgroundColor;
// We need to change these if we're active.
if(m_bActiveState == TRUE) {
int iSum =
(int)GetRValue(rgbOuter) +
(int)GetGValue(rgbOuter) +
(int)GetBValue(rgbOuter);
if(iSum > (127 * 3)) {
// Bright, use black.
rgbOuter = RGB(0, 0, 0);
}
else {
// Dark, use white.
rgbOuter = RGB(255, 255, 255);
}
}
HPEN hpO = ::CreatePen(PS_SOLID, 0, rgbOuter);
HPEN pOldPen = (HPEN)::SelectObject(hDC, hpO);
// Careful to avoid scrollers NC paints on NT and 3.1.
// We'll be detecting this by deciding how wide the window borders are,
// these systems report 1, but 95 reports 2 (and probably future versions
// of NT with a the 95 UI).
BOOL bFullPaint = FALSE;
if(sysInfo.m_iBorderWidth != 1) {
bFullPaint = TRUE;
}
// Draw left side.
::MoveToEx(hDC, crBorder.left, crBorder.top, NULL);
if(bFullPaint) {
::LineTo(hDC, crBorder.left, crBorder.bottom);
}
else if(IsHScrollBarOn()) {
::LineTo(hDC, crBorder.left, crBorder.bottom - sysInfo.m_iScrollHeight);
}
else {
::LineTo(hDC, crBorder.left, crBorder.bottom);
}
// Draw bottom side.
if(bFullPaint) {
::MoveToEx(hDC, crBorder.left, crBorder.bottom - 1, NULL);
::LineTo(hDC, crBorder.right, crBorder.bottom - 1);
}
else if(!IsHScrollBarOn()) {
::MoveToEx(hDC, crBorder.left, crBorder.bottom - 1, NULL);
if(IsVScrollBarOn()) {
::LineTo(hDC, crBorder.right - sysInfo.m_iScrollWidth, crBorder.bottom - 1);
}
else {
::LineTo(hDC, crBorder.right, crBorder.bottom - 1);
}
}
else if(IsHScrollBarOn() && IsVScrollBarOn()) {
::MoveToEx(hDC, crBorder.right - sysInfo.m_iScrollWidth + 1, crBorder.bottom - 1, NULL);
::LineTo(hDC, crBorder.right, crBorder.bottom - 1);
}
// Draw right side.
if(bFullPaint) {
::MoveToEx(hDC, crBorder.right- 1, crBorder.top, NULL);
::LineTo(hDC, crBorder.right - 1, crBorder.bottom);
}
else if(!IsVScrollBarOn()) {
::MoveToEx(hDC, crBorder.right - 1, crBorder.top, NULL);
if(IsHScrollBarOn()) {
::LineTo(hDC, crBorder.right - 1, crBorder.bottom - sysInfo.m_iScrollHeight);
}
else {
::LineTo(hDC, crBorder.right - 1, crBorder.bottom);
}
}
else if(IsVScrollBarOn() && IsHScrollBarOn()) {
::MoveToEx(hDC, crBorder.right - 1, crBorder.bottom - sysInfo.m_iScrollHeight + 1, NULL);
::LineTo(hDC, crBorder.right - 1, crBorder.bottom);
}
// Draw top side.
::MoveToEx(hDC, crBorder.left, crBorder.top, NULL);
if(bFullPaint) {
::LineTo(hDC, crBorder.right, crBorder.top);
}
else if(IsVScrollBarOn()) {
::LineTo(hDC, crBorder.right - sysInfo.m_iScrollWidth, crBorder.top);
}
else {
::LineTo(hDC, crBorder.right, crBorder.top);
}
::SelectObject(hDC, pOldPen);
::DeleteObject(hpO);
// Done with the DC.
#ifndef XP_WIN16
::ReleaseDC(GetPane(), hDC);
#else
GetFrame()->GetFrameWnd()->ReleaseDC(pDC);
#endif
}
}
}
// Tells the context wether it is made active, or inactive.
// This function mainly used to give a visual indication of the currently
// selected frame cell.
void CWinCX::ActivateCX(BOOL bNowActive) {
// Update our active state.
BOOL bOldState = m_bActiveState;
m_bActiveState = bNowActive;
// In either event, see if we should update the non client area.
if(m_bActiveState != bOldState) {
OnNcPaintCX();
}
}
// Clicking is allowed again.
// Act as though the mouse moved, so that the correct cursor is put back up.
void CWinCX::EnableClicking(MWContext *pContext)
{
BOOL bReturnImmediately = FALSE;
OnMouseMoveCX(m_uMouseFlags, m_cpMMove, bReturnImmediately);
if(bReturnImmediately) {
return;
}
}
// Function to get the view's offset into the frame's top left position.
void CWinCX::GetViewOffsetInFrame(CPoint& cpRetval) const {
// Init.
cpRetval = CPoint(0, 0);
if(GetFrame()->GetFrameWnd() != NULL && GetPane() != NULL) {
CRect cr;
::GetWindowRect(GetPane(), cr);
GetFrame()->GetFrameWnd()->ScreenToClient(cr);
cpRetval = CPoint(cr.left, cr.top);
}
}
// Attempt to act modal over the other context.
// This is assumming that this is being called from FE_MakeNewWindow after
// the popup window is created as a modal dialog.
void CWinCX::GoModal(MWContext *pParent)
{
// See if the other context can be told to disable it's UI.
if(pParent == NULL || ABSTRACTCX(pParent) == NULL ||
ABSTRACTCX(pParent)->IsWindowContext() == FALSE ||
WINCX(pParent)->GetFrame()->GetFrameWnd() == NULL) {
// No way, no UI to deactivate.
return;
}
// Okay, we can do this.
CWinCX *pDisableCX = WINCX(pParent);
// Disable their window.
// If the frame has an owned window, e.g. a modal property sheet, disable that
HWND popup = ::GetLastActivePopup(pDisableCX->GetFrame()->GetFrameWnd()->m_hWnd);
HWND popupOwner = NULL;
// Because of the order in which this takes place, the modal dialog has already been added
// in FE_MakeNewWindow, therefore it will be the last active popup. So we need to get its
// parent.
if(popup)
popupOwner = ::GetParent(popup);
if(popupOwner)
::EnableWindow(popupOwner, FALSE);
// The above call will also disable the popup since it is a child of popupOwner. Therefore
// we need to enable the popup.
::EnableWindow(popup, TRUE);
// Mark that we need to re enable the other context when we're destroyed.
// We support modality over any number of context's (in case we need to
// go modal over the application, this allows it).
if(popupOwner)
m_cplModalOver.AddTail((void *)popupOwner);
}
// Register a function to be called when the DestroyContext is called.
// We only support one at a time, so a list implementation will have to
// be provided if more than one is needed.
void CWinCX::CloseCallback(void (*pFunc)(void *), void *pArg)
{
// Just assign them in.
if(pFunc) {
m_cplCloseCallbacks.AddTail((void *)pFunc);
m_cplCloseCallbackArgs.AddTail(pArg);
}
}
// This simply tells the frame if it should do anything special in it's min max handler.
// If you really need each context to not resize (such as in Frames), then you'll have to
// redisign this.
void CWinCX::EnableResize(BOOL bEnable)
{
// Must have a frame, a CGenericFrame.
if(GetFrame()->GetFrameWnd() != NULL &&
GetFrame()->GetFrameWnd()->IsKindOf(RUNTIME_CLASS(CGenericFrame))) {
CGenericFrame *pFrame = (CGenericFrame *)GetFrame()->GetFrameWnd();
pFrame->EnableResize(bEnable);
}
}
// Don't allow the frame to close.
// This can't really work on a view by view basis (frames).
// Also, this assumes the CWinCX resides in a CFrameWnd, which will
// before handling close call CanCloseDocument on the document.
void CWinCX::EnableClose(BOOL bEnable)
{
// Get our document, and tell it wether or not it will allow the frame to close.
if(GetDocument()) {
GetDocument()->EnableClose(bEnable);
}
}
void CWinCX::DisableHotkeys(BOOL bDisable)
{
// Must have a frame, a CGenericFrame.
if(GetFrame()->GetFrameWnd() != NULL &&
GetFrame()->GetFrameWnd()->IsKindOf(RUNTIME_CLASS(CGenericFrame))) {
CGenericFrame *pFrame = (CGenericFrame *)GetFrame()->GetFrameWnd();
pFrame->DisableHotkeys(bDisable);
}
}
// Do not allow window to change z-ordering
void CWinCX::SetZOrder(BOOL bZLock, BOOL bBottommost)
{
// Must have a frame, a CGenericFrame.
if(GetFrame()->GetFrameWnd() != NULL &&
GetFrame()->GetFrameWnd()->IsKindOf(RUNTIME_CLASS(CGenericFrame))) {
CGenericFrame *pFrame = (CGenericFrame *)GetFrame()->GetFrameWnd();
pFrame->SetZOrder(bZLock, bBottommost);
}
}
// Determine wether or not we can upload files.
BOOL CWinCX::CanUploadFile()
{
BOOL bRetval = TRUE;
if(IsDestroyed()) {
bRetval = FALSE;
}
else if(GetPane() == NULL) {
bRetval = FALSE;
}
else if(CanCreateUrlFromHist() == FALSE) {
bRetval = FALSE;
}
else {
// Must check to make sure we're on the correct type of FTP
// directory.
History_entry *pHistoryEntry = SHIST_GetCurrent(&(GetContext()->hist));
if(pHistoryEntry == NULL || pHistoryEntry->address == NULL) {
bRetval = FALSE;
}
else if(strnicmp(pHistoryEntry->address, "ftp://", 6)) {
bRetval = FALSE;
}
else if(pHistoryEntry->address[strlen(pHistoryEntry->address) - 1] != '/') {
bRetval = FALSE;
}
}
return(bRetval);
}
// Upload files.
void CWinCX::UploadFile()
{
if(CanUploadFile() == FALSE) {
return;
}
// Get the files from the user.
char *pFileName = wfe_GetExistingFileName(GetPane(), szLoadString(IDS_FILE_UPLOAD), HTM, TRUE);
// Null if user hit cancel.
if(pFileName != NULL) {
URL_Struct *pUrl = CreateUrlFromHist(TRUE);
if(pUrl != NULL) {
// Set this to super reload (per montulli).
pUrl->force_reload = NET_SUPER_RELOAD;
size_t stAlloc = 2 * sizeof(char *);
pUrl->files_to_post = (char **)XP_ALLOC(stAlloc);
if(pUrl->files_to_post != NULL) {
// Clear the array.
memset(pUrl->files_to_post, 0, stAlloc);
// Put in the name.
// It's already been allocated by the prompt routine, no need to free
// it in this case.
pUrl->files_to_post[0] = pFileName;
// Ask for it.
GetUrl(pUrl, FO_CACHE_AND_PRESENT);
}
else {
NET_FreeURLStruct(pUrl);
}
}
}
}
typedef struct mouse_over_closure{
int32 xVal, yVal;
BOOL bDeleteLO_Element; // in case we created a fake one
CWinCX * pWin;
} mouse_over_closure;
//
// The backend just had a chance at a mouse-over event. If it
// didn't set the status bar text take care of setting it now
//
static void
mouse_over_callback(MWContext * context, LO_Element * lo_element, int32 event,
void * pObj, ETEventStatus status)
{
// keep track of what we have done already so that we
// don't thrash
BOOL bTextSet = FALSE;
BOOL bCursorSet = FALSE;
LO_EmbedStruct *pEmbed = NULL;
LO_ImageStruct * image_struct = NULL;
LO_TextStruct * text_struct = NULL;
// get our stashed data
mouse_over_closure * pClose = (mouse_over_closure *) pObj;
CWinCX * pWin = pClose->pWin;
// make sure the document didn't disappear
if(status == EVENT_PANIC) {
XP_FREE(pClose);
return;
}
// if the status is OK then that means the backend set the
// status bar text
if(status == EVENT_OK)
bTextSet = TRUE;
CPoint DocPoint(CASTINT(pClose->xVal), CASTINT(pClose->yVal));
BOOL bMouseInSelection = pClose->pWin->PtInSelectedRegion( DocPoint );
#ifdef EDITOR
// First check for Table mouse-over events - we don't need the current element
// since we will look for closest table hit region
// Don't do this if ALT key is pressed - this allows sizing images (and other objects)
// which are tightly surrounded by a cell boundary.
// TODO: Alt key leaves menu select mode on - how do we defeat that behavior?
if( EDT_IS_EDITOR(context) && !EDT_IsSizing(context) )
{
ED_HitType iTableHit = ED_HIT_NONE;
if( pWin->m_bSelectingCells )
{
// User is dragging mouse to select table cells
// Return value tells us what kind of cursor to use
iTableHit = EDT_ExtendTableCellSelection(context, pClose->xVal, pClose->yVal);
}
else if( GetAsyncKeyState(VK_MENU) >= 0 ) // Alt key test
{
// See if we are over a table or cell "hit" region and need a special cursor
// Get hit region - we don't need to know element, so use NULL for 4th param
// If 5th param is TRUE, then select all cells if in upper left corner of table,
// or select/unselect cell if mouse is ANYWHERE inside of the cell
iTableHit = EDT_GetTableHitRegion(context, pClose->xVal, pClose->yVal, NULL, GetAsyncKeyState(VK_CONTROL));
}
if( iTableHit )
{
bCursorSet = TRUE;
switch( iTableHit )
{
case ED_HIT_SEL_TABLE: // Upper left corner
SetCursor(theApp.LoadCursor(IDC_TABLE_SEL));
break;
case ED_HIT_SEL_ALL_CELLS: // Upper left corner with Ctrl pressed
SetCursor(theApp.LoadCursor(IDC_ALL_CELLS_SEL));
break;
case ED_HIT_SEL_COL: // Near Top table border
SetCursor(theApp.LoadCursor(IDC_COL_SEL));
break;
case ED_HIT_SEL_ROW: // Near left table border
SetCursor(theApp.LoadCursor(IDC_ROW_SEL));
break;
case ED_HIT_SEL_CELL: // Not sure - remaining cell border not matching other regions
SetCursor(theApp.LoadCursor(IDC_CELL_SEL));
break;
case ED_HIT_SIZE_TABLE_WIDTH: // Right edge of table
SetCursor(theApp.LoadCursor(IDC_TABLE_SIZE));
break;
case ED_HIT_SIZE_TABLE_HEIGHT: // Bottom edge of table
SetCursor(theApp.LoadCursor(IDC_TABLE_VSIZE));
break;
case ED_HIT_SIZE_COL: // Right border of a cell
SetCursor(theApp.LoadCursor(IDC_COL_SIZE));
break;
case ED_HIT_SIZE_ROW: // Right border of a cell
SetCursor(theApp.LoadCursor(IDC_ROW_SIZE));
break;
case ED_HIT_ADD_ROWS: // Lower left corner
SetCursor(theApp.LoadCursor(IDC_ADD_ROWS));
break;
case ED_HIT_ADD_COLS: // Lower right corner
SetCursor(theApp.LoadCursor(IDC_ADD_COLS));
break;
case ED_HIT_DRAG_TABLE: // Near bottom
SetCursor(theApp.LoadCursor(IDC_ARROW_HAND));
break;
}
goto FINISH_MOUSE_OVER;
}
}
#endif
// See if we are over some text that is a link
text_struct = (LO_TextStruct *) lo_element;
// Imagemaps send their mouseover events as dummy text structs. Be very careful
// about what you try to use from this text struct as many of the fields are
// invalid.
if(text_struct && text_struct->type == LO_TEXT && text_struct->text)
{
BOOL bIsLink = (text_struct->anchor_href && text_struct->anchor_href->anchor);
if(bIsLink && !bTextSet)
{
wfe_Progress(context, (char *) text_struct->anchor_href->anchor);
bTextSet = TRUE;
}
if( EDT_IS_EDITOR(context) )
{
#ifdef EDITOR
if( EDT_CanPasteStyle(context) )
{
SetCursor(theApp.LoadCursor(IDC_COPY_STYLE));
bCursorSet = TRUE;
}
else if( bMouseInSelection )
{
// Indicate that user can drag selected text with special cursor
SetCursor(theApp.LoadCursor(IDC_IBEAM_HAND));
bCursorSet = TRUE;
}
#else
XP_ASSERT(0);
#endif
} else if( bIsLink )
{
// Set anchor cursor only if not in editor
SetCursor(theApp.LoadCursor(IDC_SELECTANCHOR));
bCursorSet = TRUE;
}
if( !bCursorSet )
{
// We have 2 text cursor sizes -- more visible for large fonts
SetCursor( theApp.LoadCursor(
text_struct->height > 20 ? IDC_EDIT_IBEAM_LARGE : IDC_EDIT_IBEAM) );
bCursorSet = TRUE;
}
}
#ifdef EDITOR
else if( EDT_IS_EDITOR(context) &&
lo_element &&
!EDT_IsSizing(context) )
{
// Test for proximity to border of sizeable objects
// This sets the cursor to system sizing cursor
// Note: X, Y are in View coordinates
int sizing = EDT_CanSizeObject(context, lo_element, pClose->xVal, pClose->yVal);
if( sizing )
{
switch ( sizing )
{
case ED_SIZE_TOP:
case ED_SIZE_BOTTOM:
SetCursor(theApp.LoadStandardCursor(IDC_SIZENS));
break;
case ED_SIZE_RIGHT:
// Tables and cells can only size from the right border
if( lo_element->type == LO_TABLE )
{
SetCursor(theApp.LoadCursor(IDC_TABLE_SIZE));
break;
}
if( lo_element->type == LO_CELL )
{
SetCursor(theApp.LoadCursor(IDC_COL_SIZE));
break;
}
// If not Table or Cell, fall through
// to set horizontal sizing cursor
case ED_SIZE_LEFT:
SetCursor(theApp.LoadStandardCursor(IDC_SIZEWE));
break;
case ED_SIZE_TOP_RIGHT:
case ED_SIZE_BOTTOM_LEFT:
SetCursor(theApp.LoadStandardCursor(IDC_SIZENESW));
break;
case ED_SIZE_TOP_LEFT:
case ED_SIZE_BOTTOM_RIGHT:
SetCursor(theApp.LoadStandardCursor(IDC_SIZENWSE));
break;
}
bCursorSet = TRUE;
}
}
#endif // EDITOR
// NOTE: We set the imagemap and status line stuff even if we set a "sizing" cursor
// See if we are over an image that is a link
image_struct = (LO_ImageStruct *) lo_element;
if(image_struct && image_struct->type == LO_IMAGE && image_struct->image_attr) {
if(image_struct->image_attr->usemap_name != NULL) {
CPoint image_offset;
image_offset.x = CASTINT(pClose->xVal - (image_struct->x + image_struct->x_offset + image_struct->border_width));
image_offset.y = CASTINT(pClose->yVal - (image_struct->y + image_struct->y_offset + image_struct->border_width));
LO_AnchorData *pAnchorData = LO_MapXYToAreaAnchor(pWin->GetDocumentContext(), image_struct,
image_offset.x, image_offset.y);
if(pAnchorData != NULL) {
if( !EDT_IS_EDITOR(context) && !bMouseInSelection ){
SetCursor(theApp.LoadCursor(IDC_SELECTANCHOR));
bCursorSet = TRUE;
}
if( pAnchorData->anchor && !bTextSet ) {
wfe_Progress(context, (char *) pAnchorData->anchor);
bTextSet = TRUE;
}
}
}
else if((image_struct->image_attr->attrmask & LO_ATTR_ISMAP) && image_struct->anchor_href) {
// if its an ismap print the coordinates too
char buf[32];
CPoint point;
point.x = (int) (pClose->xVal - (image_struct->x + image_struct->x_offset));
point.y = (int) (pClose->yVal - (image_struct->y + image_struct->y_offset));
sprintf(buf, "?%d,%d", point.x, point.y);
CString anchor;
anchor = (char *) image_struct->anchor_href->anchor;
anchor += buf;
if( !EDT_IS_EDITOR(context) && !bMouseInSelection ){
SetCursor(theApp.LoadCursor(IDC_SELECTANCHOR));
bCursorSet = TRUE;
}
// Set progress text if mocha allows it.
if(!bTextSet) {
wfe_Progress(context, (const char *)anchor);
bTextSet = TRUE;
}
}
else if(image_struct->anchor_href) {
// Set anchor cursor only if not in editor and not in a selection
if( !EDT_IS_EDITOR(context) && !bMouseInSelection ) {
SetCursor(theApp.LoadCursor(IDC_SELECTANCHOR));
bCursorSet = TRUE;
}
if (image_struct->anchor_href->anchor && !bTextSet) {
wfe_Progress(context, (char *) image_struct->anchor_href->anchor);
bTextSet = TRUE;
}
}
else if(image_struct->image_attr->attrmask & LO_ATTR_INTERNAL_IMAGE
&& EDT_IS_EDITOR(context)
&& image_struct->alt != NULL ){
char *str;
if(!bTextSet) {
PA_LOCK(str, char *, image_struct->alt);
if( str != 0 ){
wfe_Progress( context, str );
bTextSet = TRUE;
}
PA_UNLOCK(image_struct->alt);
}
}
if( EDT_IS_EDITOR(context) && !bCursorSet ){
// Experimental: Image can be dragged, so if no cursor set above,
// use Open Hand with arrow to indicate "draggability" of the image
SetCursor(theApp.LoadCursor(IDC_ARROW_HAND));
bCursorSet = TRUE;
}
}
// See if we are over an embedded item.
pEmbed = (LO_EmbedStruct *)lo_element;
if(pEmbed && pEmbed->type == LO_EMBED && pEmbed->FE_Data) {
NPEmbeddedApp *pPluginShim = (NPEmbeddedApp *)pEmbed->FE_Data;
if(pPluginShim != NULL && wfe_IsTypePlugin(pPluginShim) == FALSE) {
CNetscapeCntrItem *pItem = (CNetscapeCntrItem *)pPluginShim->fe_data;
if(pItem != NULL && pItem->m_bLoading == FALSE && pItem->m_bBroken == FALSE &&
pItem->m_bDelayed == FALSE && pItem->m_lpObject != NULL)
{
if( !bMouseInSelection ){
SetCursor(theApp.LoadCursor(IDC_ACTIVATE_EMBED));
bCursorSet = TRUE;
}
if(!bTextSet) {
CString csEmbed;
csEmbed.LoadString(IDS_ACTIVATE_EMBED_STATUS);
CString csDescrip;
pItem->GetUserType(USERCLASSTYPE_FULL, csDescrip);
csEmbed += csDescrip;
wfe_Progress(context, (char *)(const char *)csEmbed);
bTextSet = TRUE;
}
}
}
}
FINISH_MOUSE_OVER:
// If nothing set yet blank it out and make sure we have the
// normal cursor. Do we really want to reset the cursor
// here????
// We want to show link text but keep edit cursor in Editor
if(!bCursorSet){
SetCursor(theApp.LoadStandardCursor(IDC_ARROW));
}
if(!bTextSet) {
wfe_Progress(context, "");
bTextSet = TRUE;
}
// For mouse overs on anchors we may have created a fake
// LO_Element structure. If so, blow it away now
if(pClose->bDeleteLO_Element)
XP_FREE(lo_element);
XP_FREE(pClose);
// Have the mouse timer handler do some dirty work.
// Please don't return in the above code, I'd like this to get called
// in all cases with the state of the buttons set correctly.
MouseTimerData mt(context);
FEU_MouseTimer(&mt);
}
// Function to handle the details of the cursor being over an element
// and telling the back end about it. If the backend doesn't
// set the text, the text will get set in our clallback routine
void CWinCX::FireMouseOverEvent(LO_Element *pElement, int32 xVal, int32 yVal,
CL_Layer *layer)
{
mouse_over_closure * pClose = NULL;
LO_Element * pDummy;
BOOL bEventSent = FALSE;
#define CREATE_CLOSURE() { pClose = XP_NEW_ZAP(mouse_over_closure); \
pClose->xVal = xVal; pClose->yVal = yVal; \
pClose->pWin = this; }
// The mouse is over nothing, fire an out message for the last element.
if(pElement == NULL) {
FireMouseOutEvent(TRUE, TRUE, xVal, yVal, layer);
}
// If a different element, fire an out message for the last element,
// and then fire an over message for the new element.
else if(pElement != m_pLastOverElement) {
FireMouseOutEvent(TRUE, TRUE, xVal, yVal, layer);
m_pLastOverElement = pElement;
m_pLastOverAnchorData = GetAreaAnchorData(m_pLastOverElement);
// JS needs screen coords for click events.
CPoint cpScreenPoint(xVal, yVal);
ClientToScreen(GetPane(), &cpScreenPoint);
// Fire over messages.
if(m_pLastOverAnchorData)
{
CREATE_CLOSURE();
// need to make a fake wrapper for m_pLastOverAnchorData
pDummy = (LO_Element *) XP_NEW_ZAP(LO_Element);
pDummy->lo_text.type = LO_TEXT;
pDummy->lo_text.anchor_href = m_pLastOverAnchorData;
// We use the text of the element to determine if it is still
// valid later so give the dummy text struct's text a value.
if (pDummy->lo_text.anchor_href->anchor)
pDummy->lo_text.text = pDummy->lo_text.anchor_href->anchor;
// need to free fake wrapper for m_pLastOverAnchorData
pClose->bDeleteLO_Element = TRUE;
// its safe to send the event now
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOVER;
event->x = xVal;
event->y = yVal;
event->docx = xVal + CL_GetLayerXOrigin(layer);
event->docy = yVal + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), pDummy, event,
mouse_over_callback, pClose);
bEventSent = TRUE;
}
else if(m_pLastOverElement && (m_pLastOverElement->type == LO_IMAGE ||
m_pLastOverElement->type == LO_FORM_ELE ||
(m_pLastOverElement->type == LO_TEXT
&& m_pLastOverElement->lo_text.anchor_href)
#ifdef DOM
|| LO_IsWithinSpan(m_pLastOverElement)
#endif
))
{
CREATE_CLOSURE();
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOVER;
event->x = xVal;
event->y = yVal;
event->docx = xVal + CL_GetLayerXOrigin(layer);
event->docy = yVal + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), m_pLastOverElement, event,
mouse_over_callback, pClose);
bEventSent = TRUE;
}
}
// If the same element, they have possibly traversed into a new
// AnchorData AREA, fire an out message, and then fire an
// over message if so.
// DO NOT send redundant over messages.
else if(pElement == m_pLastOverElement) {
LO_AnchorData *pAnchorData = GetAreaAnchorData(pElement);
if(pAnchorData != m_pLastOverAnchorData) {
if(m_pLastOverAnchorData) {
// Only do this if there is last anchor data, or we
// fire mouse out on the element and not the data.
FireMouseOutEvent(FALSE, TRUE, xVal, yVal, layer);
}
m_pLastOverAnchorData = pAnchorData;
if(m_pLastOverAnchorData) {
// JS needs screen coords for click events.
CPoint cpScreenPoint(xVal, yVal);
ClientToScreen(GetPane(), &cpScreenPoint);
CREATE_CLOSURE();
// need to make a fake wrapper for m_pLastOverAnchorData
pDummy = (LO_Element *) XP_NEW_ZAP(LO_Element);
pDummy->lo_text.type = LO_TEXT;
pDummy->lo_text.anchor_href = m_pLastOverAnchorData;
// We use the text of the element to determine if it is still
// valid later so give the dummy text struct's text a value.
if (pDummy->lo_text.anchor_href->anchor)
pDummy->lo_text.text = pDummy->lo_text.anchor_href->anchor;
// need to free fake wrapper for m_pLastOverAnchorData
pClose->bDeleteLO_Element = TRUE;
// its safe to send the event now
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOVER;
event->x = xVal;
event->y = yVal;
event->docx = xVal + CL_GetLayerXOrigin(layer);
event->docy = yVal + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), pDummy, event,
mouse_over_callback, pClose);
bEventSent = TRUE;
}
}
// If this is an ismap and we've moved within it, we need to
// change status without firing off a mouse over.
else if(pElement->lo_any.type == LO_IMAGE) {
LO_ImageStruct *image_struct = (LO_ImageStruct *)pElement;
if((image_struct->image_attr->usemap_name == NULL) && (image_struct->image_attr->attrmask & LO_ATTR_ISMAP) && image_struct->anchor_href) {
// if its an ismap print the coordinates too
char buf[32];
CPoint point;
point.x = (int) (xVal - (image_struct->x + image_struct->x_offset));
point.y = (int) (yVal - (image_struct->y + image_struct->y_offset));
sprintf(buf, "?%d,%d", point.x, point.y);
CString anchor;
anchor = (char *) image_struct->anchor_href->anchor;
anchor += buf;
if (!EDT_IS_EDITOR(GetContext())) {
SetCursor(theApp.LoadCursor(IDC_SELECTANCHOR));
}
// Set progress text
wfe_Progress(GetContext(), (const char *)anchor);
}
if (!EDT_IS_EDITOR(GetContext()))
bEventSent = TRUE;
}
// Editor wants callback to be callled every time, the Browser does not. If
// we're over the same element it may have set the status bar message when we
// entered it. Don't let the Browser set the status bar again until we leave it.
// If the element didn't want to set the status bar, the Browser will have already
// set it correctly anyway.
else if (!EDT_IS_EDITOR(GetContext()))
bEventSent = TRUE;
}
// In the Editor, ALWAYS call the final callback
// so we can monitor mouse location for dynamic resizing of elements
if( !bEventSent ) {
CREATE_CLOSURE();
// Note: param 3 ("event") isn't used in mouse_over_callback
mouse_over_callback(GetContext(), pElement, 0, (void*)pClose, EVENT_CANCEL);
}
// We're using a timer to deal with the lack of a Windows call for entrance
// and exit for a window. Deal with it here.
if (!m_bMouseMoveTimerSet) {
// This is our first time into the document or our last. Start the timer to check
// for our exit and fire a mouseover for the window.
POINT mp;
// get mouse position
::GetCursorPos(&mp);
if (::WindowFromPoint(mp) == GetPane()) {
MouseMoveTimerData mt(GetContext());
FEU_MouseMoveTimer(&mt);
}
}
/* Unix and Mac are not up to spec on these and won't be for 4.0 Commenting
* them out for future use later */
/*JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOVER;
event->x = xVal;
event->y = yVal;
event->layer_id = LO_DOCUMENT_LAYER_ID;
ET_SendEvent(GetContext(), 0, event, 0, 0);
}
else {
// We may have left the document. If so, send a mouseout to the window.
POINT mp;
// get mouse position
::GetCursorPos(&mp);
if ((::WindowFromPoint(mp) != GetPane()) && !m_bLBDown) {
//Yup, we left.
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOUT;
event->x = xVal;
event->y = yVal;
event->layer_id = LO_DOCUMENT_LAYER_ID;
ET_SendEvent(GetContext(), 0, event, 0, 0);
}
} */
return;
#undef CREATE_CLOSURE
}
// Retrieve anchor data out of areas only (usemaps)
// Use last known mouse move coordinates to do so.
LO_AnchorData *CWinCX::GetAreaAnchorData(LO_Element *pElement) {
LO_AnchorData *pRetval = NULL;
// Make sure this is an image element.
if(pElement && pElement->lo_any.type == LO_IMAGE) {
// Determine coordinates in pixels with image as origin and
// ask layout for area anchor data, will return NULL if none.
LTRB Rect;
LO_ImageStruct *pLOImage = (LO_ImageStruct *)pElement;
// Use the returned value to convert the mouse coordinates
// to be image relative.
ResolveElement(Rect, pLOImage->x, pLOImage->y,
pLOImage->x_offset + pLOImage->border_width,
pLOImage->y_offset + pLOImage->border_width,
0, 0);
if (pLOImage->layer) {
CL_Layer *parent;
int32 x_offset, y_offset;
parent = CL_GetLayerParent(pLOImage->layer);
if (parent) {
x_offset = CL_GetLayerXOrigin(parent);
y_offset = CL_GetLayerYOrigin(parent);
Rect.left += x_offset;
Rect.right += x_offset;
Rect.top += y_offset;
Rect.bottom += y_offset;
}
}
pRetval = LO_MapXYToAreaAnchor(GetDocumentContext(), pLOImage,
m_cpMMove.x - Rect.left, m_cpMMove.y - Rect.top);
}
return(pRetval);
}
static void
free_this_callback(MWContext * context, LO_Element * lo_element, int32 event,
void * pObj, ETEventStatus status)
{
XP_FREE(pObj);
}
// Fire mouse out events for cached element types.
void CWinCX::FireMouseOutEvent(BOOL bClearElement, BOOL bClearAnchor, int32 xVal,
int32 yVal, CL_Layer *layer)
{
// JS needs screen coords for click events.
CPoint cpScreenPoint(xVal, yVal);
ClientToScreen(GetPane(), &cpScreenPoint);
// Determine if we're sending an anchor out or an element out.
if(m_pLastOverAnchorData)
{
// Anchor data.
// Create a fake holder element here since we didn't save enough
// information and Win16 won't let us pass stack variables
LO_Element * element = XP_NEW_ZAP(LO_Element);
element->lo_text.type = LO_TEXT;
element->lo_text.anchor_href = m_pLastOverAnchorData;
// We use the text of the element to determine if it is still
// valid later so give the dummy text struct's text a value.
if (element->lo_text.anchor_href->anchor)
element->lo_text.text = element->lo_text.anchor_href->anchor;
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOUT;
event->x = xVal;
event->y = yVal;
event->docx = xVal + CL_GetLayerXOrigin(layer);
event->docy = yVal + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), element, event,
free_this_callback, element);
}
else if(m_pLastOverElement && (m_pLastOverElement->type == LO_IMAGE
|| m_pLastOverElement->type == LO_FORM_ELE ||
(m_pLastOverElement->type == LO_TEXT &&
m_pLastOverElement->lo_text.anchor_href)
#ifdef DOM
|| LO_IsWithinSpan(m_pLastOverElement)
#endif
))
{
// Element.
JSEvent *event;
event = XP_NEW_ZAP(JSEvent);
event->type = EVENT_MOUSEOUT;
event->x = xVal;
event->y = yVal;
event->docx = xVal + CL_GetLayerXOrigin(layer);
event->docy = yVal + CL_GetLayerYOrigin(layer);
event->screenx = cpScreenPoint.x;
event->screeny = cpScreenPoint.y;
event->layer_id = LO_GetIdFromLayer(GetContext(), layer);
ET_SendEvent(GetContext(), m_pLastOverElement, event,
NULL, this);
}
//
// Mocha may have trashed the docuemnt, but it will still be safe
// to set the following elements to NULL. Mocha can't have
// destroyed our context yet
//
if(bClearElement) {
m_pLastOverElement = NULL;
}
if(bClearAnchor) {
m_pLastOverAnchorData = NULL;
}
// Clean up, these can now be empty.
m_bLastOverTextSet = FALSE;
}
CFrameGlue *CWinCX::GetFrame() const
{
// If we're a grid cell, return the
// parent's frame (one in the same)
// This allows those scary times when we're
// a grid in an OLE server to work better
// when the frame's switch.
MWContext *pParent = GetParentContext();
if(IsGridCell() && pParent) {
if(ABSTRACTCX(pParent)->IsFrameContext()) {
return(WINCX(pParent)->GetFrame());
}
}
// Make sure our frame isn't NULL.
// If so, go with the NULL frame.
// Could happen if the frame is destroyed, but the
// context lives.
if(m_pFrame == NULL) {
return(m_pNullFrame);
}
return m_pFrame;
}
void CWinCX::ClearView()
{
// Tell the view and frame that we're no more.
if(m_pGenView != NULL) {
// Also, if we're selecting text, clear our capture of the mouse.
if(m_bLBDown == TRUE) {
::ReleaseCapture();
}
m_pGenView->ClearContext();
m_pGenView = NULL;
}
}
BOOL CWinCX::IsClickingEnabled() const
{
#ifdef EDITOR
return(GetContext()->waitingMode == FALSE && GetContext()->edit_saving_url == FALSE);
#else
return(GetContext()->waitingMode == FALSE);
#endif
}
//#ifndef NO_TAB_NAVIGATION
LO_TabFocusData * CWinCX::getLastTabFocusData()
{
return( &m_lastTabFocus );
}
LO_Element * CWinCX::getLastTabFocusElement()
{
return( m_lastTabFocus.pElement );
}
int32 CWinCX::getLastFocusAreaIndex()
{
return( m_lastTabFocus.mapAreaIndex ); // 0 means no focus, start with index 1.
}
LO_AnchorData *CWinCX::getLastFocusAnchorPointer()
{
return( m_lastTabFocus.pAnchor );
}
char *CWinCX::getLastFocusAnchorStr()
{
if( m_lastTabFocus.pAnchor == NULL)
return( NULL );
return( (char *)m_lastTabFocus.pAnchor->anchor );
}
// When an element gets or losts Tab Focus, we need to invalidate it and hence to redraw it.
// check if(GetPane() && CanBlockDisplay() ) before calling invalidateElement().
void CWinCX::invalidateElement( LO_Element *pElement )
{
if( pElement == NULL )
return;
// find out which element needs refresh.
LTRB Rect;
BOOL rectEmpty;
LO_ImageStruct *pImage;
LO_Any *pp = (LO_Any *)pElement;
rectEmpty = TRUE;
switch( pElement->lo_any.type ) {
case LO_FORM_ELE :
if( LO_isFormElementNeedTextTabFocus( (LO_FormElementStruct *)pElement) ) {
// for Form element type FORM_TYPE_RADIO and FORM_TYPE_CHECKBOX,
// it is the next Text element, with dotted box, needs refresh.
pp = (LO_Any *)pElement->lo_any.next;
}
// for other Form element type( button, text input...), the widget handles the focus.
ResolveElement(Rect, pp->x, pp->y, pp->x_offset,pp->y_offset, pp->width, pp->height);
rectEmpty = FALSE;
break;
case LO_TEXT :
// for a text(link), the element itself has a dotted box as focus.
ResolveElement(Rect, pp->x, pp->y, pp->x_offset,pp->y_offset, pp->width, pp->height);
rectEmpty = FALSE;
break;
case LO_IMAGE :
// TODO only invalidate the area, not the whole map image.
pImage = &pElement->lo_image;
ResolveElement(Rect, IL_GetImagePixmap(pImage->image_req),
pImage->x_offset + pImage->border_width,
pImage->y_offset + pImage->border_width,
pImage->x, pImage->y, pImage->width, pImage->height);
rectEmpty = FALSE;
break;
default :
rectEmpty = TRUE;
break;
}
if( rectEmpty )
return;
// no background erase
::InvalidateRect(GetPane(), CRect(CASTINT(Rect.left), CASTINT(Rect.top), CASTINT(Rect.right), CASTINT(Rect.bottom)), FALSE);
}
void CWinCX::SetActiveWindow()
{
// Set my view to be the active view, so I will get keyboard input.
// SetFocus cannot get keyboard, and it would grab focus from
// Form elements. They need focus to process SpaceBar etc.
CFrameGlue * pFrame = GetFrame();
if( pFrame ) {
CFrameWnd * pFrameWindow = pFrame->GetFrameWnd();
if( pFrameWindow && pFrameWindow->IsKindOf(RUNTIME_CLASS(CGenericFrame))) {
CGenericView *pView = GetView();
if( pView )
pFrameWindow->SetActiveView( (CView *)pView );
}
} // else something wrong.
}
// this function is trigered by clicking in a form element, or link,
// the form element has set focus to itself. we need to clear old focus only.
// for clicking on link, we don't need to call Windows' setFocus().
void CWinCX::setFormElementTabFocus( LO_Element * pFormElement )
{
LO_TabFocusData newTabFocus;
if(pFormElement == getLastTabFocusElement() ) // clicking on the focused element.
return;
newTabFocus.pElement = pFormElement;
MWContext *pContext = GetContext();
// for clicking area.
newTabFocus.mapAreaIndex = 0; // 0 means no area.
newTabFocus.pAnchor = NULL;
// double check tab-able, and fill in pAnchor
if( LO_isTabableElement(pContext, &newTabFocus ) ) {
setLastTabFocusElement( &newTabFocus, 0 ); // clicked, 0 means don't needSetFocus
SetMainFrmTabFocusFlag(CMainFrame::TAB_FOCUS_IN_GRID); // I have tab focus.
}
}
// text element may be fragmented in multiple lines, for Tab_focus, they
// need to be treated as one tab stop.
int CWinCX::invalidateSegmentedTextElement(LO_TabFocusData *pNextTabFocus, int forward )
{
int count = 0;
LO_Element *pElement;
pElement = pNextTabFocus->pElement;
while( pElement &&
(pElement->lo_any.type == LO_TEXT || pElement->lo_any.type == LO_LINEFEED) ) {
// skip LO_LINEFEED
if( pElement->lo_any.type == LO_TEXT ) {
if( pElement->lo_text.anchor_href
&& pElement->lo_text.anchor_href == pNextTabFocus->pAnchor ) {
FEU_MakeElementVisible( GetContext(), (LO_Any *) pElement );
count++;
invalidateElement( pElement );
if( ! forward )
pNextTabFocus->pElement = pElement; // move the current focus pointer to the first segment.
} else {
break; // stop searching ;
}
} // if( pElement->lo_any.type == LO_TEXT )
if( forward)
pElement = pElement->lo_any.next;
else
pElement = pElement->lo_any.prev;
} // while( pElement &&
return count;
}
// this function is trigered by TAB key
// or called from setFormElementTabFocus() with byClickFormElement = 0, in such case
// the form element is visible and has the fucos already.
void CWinCX::setLastTabFocusElement( LO_TabFocusData *pNextTabFocus, int needSetFocus )
{
LO_Element *pElement;
// both old and new element can be NULL.
// avoid setting focus on the same element.
// For image map, the element may be the same, but the mapAreaIndex
// increase for each Tab.
if( pNextTabFocus
&& m_lastTabFocus.pElement == pNextTabFocus->pElement
&& m_lastTabFocus.mapAreaIndex == pNextTabFocus->mapAreaIndex )
return;
if( m_isReEntry_setLastTabFocusElement )
return;
m_isReEntry_setLastTabFocusElement = 1;
// need to redraw those 2 elements for visual feedback.
if(GetPane() && CanBlockDisplay() ) {
// the new element gets the focus.
pElement = pNextTabFocus->pElement;
if( NULL != pElement) {
if( needSetFocus ) {
SetActiveWindow();
// FEU_MakeElementVisible must happen before invalidate
FEU_MakeElementVisible( GetContext(), (LO_Any *) pElement );
invalidateElement( pElement );
if( pElement->lo_any.type == LO_FORM_ELE) {
// Form elements need focus to get keyboard input.
// this will loop back to setFormElementTabFocus(), That
// is why we need the re-entry protection.
FE_FocusInputElement(GetContext(), pElement );
} else {
CGenericFrame * pFrame = (CGenericFrame * )FEU_GetLastActiveFrame();
if( pFrame )
pFrame->SetFocus(); // set focus to the window.
}
}
if( pElement->lo_any.type == LO_TEXT // defined_as 1
|| pElement->lo_any.type == LO_FORM_ELE // defined_as 6
&& LO_isFormElementNeedTextTabFocus( (LO_FormElementStruct *)pElement) ) {
// first trace back for the first segmented text.
// need to do this when: tab backward, and click on not-first segment.
invalidateSegmentedTextElement(pNextTabFocus, 1); // forward
// trace down the fragmented text for a link.
invalidateSegmentedTextElement(pNextTabFocus, 0); // backward, set to first
}
} // if( NULL != pElement) , the new element
// the old element losts the focus
pElement = getLastTabFocusElement();
if( NULL != pElement) {
invalidateElement( pElement );
if( pElement->lo_any.type == LO_FORM_ELE) {
FE_BlurInputElement(GetContext(), pElement );
}
if( pElement->lo_any.type == LO_TEXT
|| pElement->lo_any.type == LO_FORM_ELE
&& LO_isFormElementNeedTextTabFocus( (LO_FormElementStruct *)pElement) ) {
// remove focus box for all continnuios text fragments.
// because it is always pointing to the first segment, no need to go backward.
invalidateSegmentedTextElement(getLastTabFocusData(), 1); // forward
}
} // if( NULL != pElement), the old element
} // if(GetPane() && CanBlockDisplay() )
m_lastTabFocus.pElement = pNextTabFocus->pElement;
m_lastTabFocus.mapAreaIndex = pNextTabFocus->mapAreaIndex; // 0 means no focus, start with index 1.
m_lastTabFocus.pAnchor = pNextTabFocus->pAnchor ;
// update the status bar.
// For all LO_ types which update status bar, see CWinCX::OnMouseMoveForLayerCX(UINT uFlags, CPoint& cpPoint,
if( getLastFocusAnchorStr() != NULL )
wfe_Progress(GetContext(), getLastFocusAnchorStr() );
else
wfe_Progress(GetContext(), "");
m_isReEntry_setLastTabFocusElement = 0;
return;
} // CWinCX::setLastTabFocusElement()
// try to set Tab Focus in this CWinCX only.
BOOL CWinCX::setNextTabFocusInWin( int forward )
{
BOOL found;
LO_TabFocusData newTabFocus;
newTabFocus.pElement = m_lastTabFocus.pElement;
newTabFocus.mapAreaIndex = m_lastTabFocus.mapAreaIndex; // 0 means no area
newTabFocus.pAnchor = m_lastTabFocus.pAnchor ;
found = LO_getNextTabableElement( GetContext(), &newTabFocus, forward );
// even new element is NULL, need to clear the old focus
setLastTabFocusElement( &newTabFocus, 1 ); // key, not click, needSetFocus
return( found );
} // BOOL CWinCX::setNextTabFocusInWin( int forward )
// this function searchs for sibling CWinCX, and set Tab focus.
// See MWContext * XP_FindNamedContextInList(MWContext * context, char *name)
// in ns\lib\xp\xp_cntxt.c for navigating grid_parent - grid_children tree.
BOOL CWinCX::setTabFocusNext( int forward )
{
BOOL ret;
// First try myself and my children.
ret = setTabFocusNextChild( NULL, forward ) ; // no exclusive child
if( ret )
return( TRUE ); // all done
// TODO loop to search up all parents
// now try to set focus to siblings
// if I am on the top, I cannot have sibling.
if( IsGridCell() != TRUE )
return( FALSE );
MWContext *pParent = GetParentContext();
if( pParent == NULL)
return( FALSE );
if( ! ABSTRACTCX(pParent)->IsWindowContext() )
return( FALSE );
ret = WINCX(pParent)->setTabFocusNextChild( GetContext(), forward ) ;
if ( ret )
return( TRUE );
return( FALSE );
}
BOOL CWinCX::setTabFocusNextChild( MWContext *currentChildContext, int forward )
{
BOOL ret;
XP_List *gridListHead, *gridList;
MWContext *pChild;
// MWContext *pCurrent = ( MWContext *) currentChild;
// try myself first
ret = setNextTabFocusInWin( forward ) ;
if( ret ) {
return( TRUE );
}
// If not a parent, return. over kill?
if( IsGridParent() == FALSE)
return( FALSE );
// see ns\lib\xp\xp_list.c for navigating the list.
gridListHead = GetContext()->grid_children;
if( currentChildContext )
gridList = XP_ListFindObject( gridListHead, currentChildContext );
else
gridList = gridListHead; // , start from head, The head node is always empty(no data).
if( gridList != NULL ) // revers order??
gridList = forward ? gridList->prev : gridList->next ; // the list is in revers order
while( gridList != NULL ) {
pChild = (MWContext *) gridList->object ;
if( ABSTRACTCX(pChild)->IsWindowContext() ) {
TRACE0("recursive setTabFocusNextChild()\n");
ret = WINCX(pChild)->setTabFocusNextChild( NULL, forward ) ; // no exlusive child.
if( ret )
return( ret );
}
gridList = forward ? gridList->prev : gridList->next ; // the list is in revers order
}
return( FALSE );
}
BOOL CWinCX::fireTabFocusElement( UINT nChar)
{
int32 mapAreaIndex, xx, yy;
lo_MapAreaRec *theArea;
LO_Element *pFocusElement = getLastTabFocusElement();
if( pFocusElement == NULL )
return( FALSE );
// SpaceBar Return are handled by Form itself for
// all Form elements, including checkbox and radio.
// Here we only take care of the links
if( pFocusElement->lo_any.type == LO_IMAGE ) {
mapAreaIndex = getLastFocusAreaIndex();
if( mapAreaIndex > 0 ) {
// the focus is in a map area
theArea = LO_getTabableMapArea( GetContext(), (LO_ImageStruct *)pFocusElement, mapAreaIndex );
if( theArea && LO_findApointInArea( theArea, &xx, &yy ) ) {
// Area coordinates are relative to the image, add the left-top of the image.
xx += pFocusElement->lo_any.x;
yy += pFocusElement->lo_any.y;
if ( FE_ClickAnyElement( GetContext(), pFocusElement, 1, xx, yy ) ) // click the center
return( TRUE );
}
return( FALSE );
}
// else mapAreaIndex not > 0, it is a link for the whole image, fall through.
}
if( pFocusElement->lo_any.type == LO_TEXT
|| pFocusElement->lo_any.type == LO_IMAGE ) {
// it is a link
if ( FE_ClickAnyElement( GetContext(), pFocusElement, 0, 0, 0 ) ) // click the center
return( TRUE );
}
return( FALSE );
}
int CWinCX::getImageDrawFlag( MWContext *pContext, LO_ImageStruct *pImage, lo_MapAreaRec **ppArea, uint32 *pFlag )
{
LO_Element *pFocusElement;
int32 lastAreaIndex;
lo_MapAreaRec *theArea;
// clear the flag first
*pFlag &= ~FE_DRAW_TAB_FOCUS ;
if( pImage == NULL )
return( 0 );
pFocusElement = getLastTabFocusElement();
if( pFocusElement == NULL)
return( 0 );
if( pFocusElement != (LO_Element *) pImage )
return( 0 );
lastAreaIndex = getLastFocusAreaIndex();
if( lastAreaIndex < 0 ) // error
return(0);
if( lastAreaIndex == 0 ) { // no area, the whole image is focused.
LO_AnchorData *pLastAnchor = getLastFocusAnchorPointer();
if (pLastAnchor && !(pLastAnchor->flags & ANCHOR_SUPPRESS_FEEDBACK)) {
*pFlag |= FE_DRAW_TAB_FOCUS;
return(1);
}
}
theArea = LO_getTabableMapArea( pContext, pImage, lastAreaIndex );
// Don't draw selection rectangle if it's been disabled
if(theArea && !(theArea->anchor && (theArea->anchor->flags & ANCHOR_SUPPRESS_FEEDBACK))) {
*pFlag |= FE_DRAW_TAB_FOCUS ;
*ppArea = theArea; // only the area is focused.
return( 1 );
}
return( 0 );
}
int CWinCX::setTextTabFocusDrawFlag( LO_TextStruct *pText, uint32 *pFlag )
{
// Visual feedback for Tab Focus is a dotted box around text or image.
// For 2 Form elements, check box and radio button, the box is on
// the text following the button.
//
// The text may have been fragmented into multiple lines because the line folding.
// When line width changes, fragments can be generated, or merged. It can happen
// anytime when the width of Navigator is changed. So, it cannot be handled
// when Tab key is pressed.
//
// When searching for next Tabable element, focus is moved to next different anchor.
// Continuious text fragments are skipped.
// See LO_getNextTabableElement() in lib\layout\laysel.c file.
//
// To find all focus text fragments, the Anchor data is checked. Every fragment has
// its own copy of duplicated Anchor pointer. If it is the same as the Focused element,
// the text will be drawn as focused.
//
// When an element losts Tab focus, all dotted fragments need redraw to erase the focus.
//
// assuming no focus and clear the flag first.
*pFlag &= ~FE_DRAW_TAB_FOCUS ;
if( pText == NULL )
return(0);
LO_Element *pCurrentElement, *pPreviousElement, *pFocusElement;
pFocusElement = getLastTabFocusElement();
if( pFocusElement == NULL)
return(0);
// Don't draw selection rectangle if it's been disabled
if ( pText->anchor_href
&& pText->anchor_href->flags & ANCHOR_SUPPRESS_FEEDBACK)
return(0);
pCurrentElement = (LO_Element *) pText;
// return focus if it is a link and it has Tab focus
if( pFocusElement == pCurrentElement ) {
*pFlag |= FE_DRAW_TAB_FOCUS;
return( 1 );
}
// test fragmented text for a link.
// is one fragmented element.
if( pFocusElement->lo_any.type == LO_TEXT // fix for bug #45111
&& pText->anchor_href
&& pText->anchor_href == getLastFocusAnchorPointer() ) {
*pFlag |= FE_DRAW_TAB_FOCUS ;
return( 1 );
}
// Test if it is a text element following a Form element checkBox(or radio button),
// which has focus. If it is, the text(not the button) needs to draw the focus box.
//todo in html, two separated element may have the same link. need to make sure it
pPreviousElement = pCurrentElement->lo_any.prev;
if( pPreviousElement != NULL && pPreviousElement == pFocusElement ) {
if( LO_isFormElementNeedTextTabFocus( (LO_FormElementStruct *)pPreviousElement ) ) {
*pFlag |= FE_DRAW_TAB_FOCUS ;
return(1);
}
}
return(0);
} // isTabFocusText()
//#endif /* NO_TAB_NAVIGATION */
void CWinCX::CancelSizing()
{
#ifdef EDITOR
ASSERT(EDT_IS_EDITOR(GetContext()));
// Remove last sizing feedback
DisplaySelectionFeedback(LO_ELE_SELECTED, m_rectSizing);
EDT_CancelSizing(GetContext());
#endif // EDITOR
}