зеркало из https://github.com/mozilla/pjs.git
2742 строки
73 KiB
C++
2742 строки
73 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
|
|
*
|
|
* 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 "usertlbr.h"
|
|
#include "shcut.h"
|
|
#include "dropmenu.h"
|
|
#include "prefapi.h"
|
|
#include "xp_ncent.h"
|
|
#include "rdfliner.h"
|
|
#include "urlbar.h"
|
|
#include "intlwin.h"
|
|
#include "libi18n.h"
|
|
|
|
extern "C" {
|
|
#include "xpgetstr.h"
|
|
};
|
|
|
|
// The Nav Center vocab element
|
|
extern "C" RDF_NCVocab gNavCenter;
|
|
|
|
#define TEXT_CHARACTERS_SHOWN 9
|
|
#define BORDERSIZE 2
|
|
#define TEXTVERTMARGIN 1
|
|
#define TEXTONLYVERTMARGIN 2
|
|
#define BITMAPVERTMARGIN 2
|
|
#define TEXT_BITMAPVERTMARGIN 1
|
|
#define HORIZMARGINSIZE 4
|
|
|
|
#define LEFT_TOOLBAR_MARGIN 10
|
|
#define RIGHT_TOOLBAR_MARGIN 20
|
|
#define SPACE_BETWEEN_BUTTONS 2
|
|
|
|
#define MAX_TOOLBAR_BUTTONS 60
|
|
#define MAX_TOOLBAR_ROWS 3
|
|
|
|
int CRDFToolbar::m_nMinToolbarButtonChars = 15;
|
|
int CRDFToolbar::m_nMaxToolbarButtonChars = 30;
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
|
// CRDFToolbarButtonDropTarget
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
DROPEFFECT CRDFToolbarButtonDropTarget::ProcessDragEnter(CWnd *pWnd, COleDataObject *pDataObject,
|
|
DWORD dwKeyState, CPoint point)
|
|
{
|
|
|
|
// Drop target is only used for drops directly onto folder personal toolbar buttons.
|
|
// This drop target is NOT used for the Aurora selector bar drops.
|
|
return ProcessDragOver(pWnd, pDataObject, dwKeyState, point);
|
|
}
|
|
|
|
DROPEFFECT CRDFToolbarButtonDropTarget::ProcessDragOver(CWnd *pWnd, COleDataObject *pDataObject,
|
|
DWORD dwKeyState, CPoint point)
|
|
{
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)m_pButton;
|
|
|
|
// Treat as a drop onto the folder.
|
|
return RDFGLOBAL_TranslateDropAction(pButton->GetNode(), pDataObject, 2);
|
|
}
|
|
|
|
BOOL CRDFToolbarButtonDropTarget::ProcessDrop(CWnd *pWnd, COleDataObject *pDataObject,
|
|
DROPEFFECT dropEffect, CPoint point)
|
|
{
|
|
// Treat as a drop onto the folder.
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)m_pButton;
|
|
RDFGLOBAL_PerformDrop(pDataObject, pButton->GetNode(), 2);
|
|
return TRUE;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
|
// CRDFToolbarButton
|
|
//////////////////////////////////////////////////////////////////////////////////////
|
|
static HBITMAP m_hbmpBM = NULL;
|
|
static HBITMAP m_hbmpBMSelected = NULL;
|
|
static HBITMAP m_hbmpFolder = NULL;
|
|
static HBITMAP m_hbmpSelectedFolder = NULL;
|
|
static HBITMAP m_hbmpFolderOpen = NULL;
|
|
static int nBitmapRefCount = 0;
|
|
|
|
CRDFToolbarButton::CRDFToolbarButton()
|
|
{
|
|
m_bShouldShowRMMenu = TRUE;
|
|
|
|
m_Node = NULL;
|
|
|
|
m_uCurBMID = 0;
|
|
|
|
currentRow = 0;
|
|
|
|
m_pTreeView = NULL;
|
|
}
|
|
|
|
|
|
CRDFToolbarButton::~CRDFToolbarButton()
|
|
{
|
|
if(m_Node && HT_IsContainer(m_Node))
|
|
{
|
|
nBitmapRefCount--;
|
|
|
|
if(nBitmapRefCount == 0)
|
|
{
|
|
if(m_hbmpBM)
|
|
{
|
|
DeleteObject(m_hbmpBM);
|
|
}
|
|
|
|
if(m_hbmpFolder)
|
|
{
|
|
DeleteObject(m_hbmpFolder);
|
|
}
|
|
|
|
if(m_hbmpSelectedFolder)
|
|
{
|
|
DeleteObject(m_hbmpSelectedFolder);
|
|
}
|
|
|
|
if(m_hbmpFolderOpen)
|
|
{
|
|
DeleteObject(m_hbmpFolderOpen);
|
|
}
|
|
|
|
if(m_hbmpBMSelected)
|
|
{
|
|
DeleteObject(m_hbmpBMSelected);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
int CRDFToolbarButton::Create(CWnd *pParent, int nToolbarStyle, CSize noviceButtonSize, CSize advancedButtonSize,
|
|
LPCTSTR pButtonText, LPCTSTR pToolTipText,
|
|
LPCTSTR pStatusText,
|
|
CSize bitmapSize, int nMaxTextChars, int nMinTextChars, BOOKMARKITEM bookmark,
|
|
HT_Resource pNode, DWORD dwButtonStyle )
|
|
{
|
|
m_bookmark = bookmark;
|
|
|
|
int16 localecsid = CIntlWin::GetSystemLocaleCsid();
|
|
|
|
// Ad Hoc I18N Fix
|
|
// Untill we draw the status bar by ourself
|
|
// The pStatusText come in CRDFToolbarButton is always UTF8
|
|
// The pStatusText pass to CToolbarButton is the locale csid
|
|
LPSTR pLocaleStatusText = (LPSTR)
|
|
INTL_ConvertLineWithoutAutoDetect(
|
|
CS_UTF8,
|
|
localecsid,
|
|
(unsigned char*)pStatusText,
|
|
XP_STRLEN(pStatusText)
|
|
);
|
|
|
|
// Ad Hoc I18N Fix
|
|
// Untill we bring back the real implementation of
|
|
// tooltip.cpp
|
|
// The pToolTipText come in CRDFToolbarButton is always UTF8
|
|
// The pToolTipText pass to CToolbarButton is the locale csid
|
|
LPSTR pLocaleToolTipText = (LPSTR)
|
|
INTL_ConvertLineWithoutAutoDetect(
|
|
CS_UTF8,
|
|
localecsid,
|
|
(unsigned char*)pToolTipText,
|
|
XP_STRLEN(pToolTipText)
|
|
);
|
|
|
|
// For pButtonText,
|
|
// We do not convert it to the limited locale csid because we can
|
|
// Draw the text by ourself to overcome the OS limitation.
|
|
|
|
BOOL bResult = CToolbarButton::Create(pParent, nToolbarStyle, noviceButtonSize, advancedButtonSize,
|
|
pButtonText, pLocaleToolTipText, pLocaleStatusText, 0, 0,
|
|
bitmapSize, TRUE, 0, nMaxTextChars, nMinTextChars, dwButtonStyle);
|
|
|
|
XP_FREEIF(pLocaleStatusText);
|
|
XP_FREEIF(pLocaleToolTipText);
|
|
|
|
if(bResult)
|
|
{
|
|
SetNode(pNode);
|
|
UpdateIconInfo();
|
|
if (m_menu.m_hMenu == NULL || (m_menu.m_hMenu != NULL && !IsMenu(m_menu.m_hMenu)))
|
|
m_menu.CreatePopupMenu();
|
|
}
|
|
|
|
return bResult;
|
|
}
|
|
|
|
void CRDFToolbarButton::UpdateIconInfo()
|
|
{
|
|
m_nIconType = DetermineIconType(m_Node, UseLargeIcons());
|
|
UINT oldBitmapID = m_nBitmapID;
|
|
m_nBitmapID = GetBitmapID();
|
|
if (m_nBitmapID != oldBitmapID)
|
|
{
|
|
HDC hDC = ::GetDC(m_hWnd);
|
|
HPALETTE hPalette = WFE_GetUIPalette(GetParentFrame());
|
|
HBITMAP hBitmap = WFE_LookupLoadAndEnterBitmap(hDC, m_nBitmapID, TRUE, hPalette,
|
|
sysInfo.m_clrBtnFace, RGB(255, 0, 255));
|
|
::ReleaseDC(m_hWnd, hDC);
|
|
SetBitmap(hBitmap, TRUE);
|
|
if (m_nBitmapID == IDB_PICTURES)
|
|
SetBitmapSize(CSize(23,21)); // Command buttons
|
|
else SetBitmapSize(CSize(23,17)); // Personal toolbar buttons
|
|
}
|
|
|
|
m_nBitmapIndex = GetBitmapIndex();
|
|
}
|
|
|
|
CSize CRDFToolbarButton::GetMinimalButtonSize(void)
|
|
{
|
|
CString szText(HT_GetNodeName(m_Node));
|
|
return GetButtonSizeFromChars(szText, m_nMinTextChars);
|
|
}
|
|
|
|
CSize CRDFToolbarButton::GetMaximalButtonSize(void)
|
|
{
|
|
CString szText(HT_GetNodeName(m_Node));
|
|
return GetButtonSizeFromChars(szText, m_nMaxTextChars);
|
|
}
|
|
|
|
void CRDFToolbarButton::OnAction(void)
|
|
{
|
|
if(m_Node)
|
|
{
|
|
char* url = HT_GetNodeURL(m_Node);
|
|
if (strncmp(url, "command:", 8) == 0)
|
|
{
|
|
// We're a command, baby. Look up our FE command and execute it.
|
|
UINT nCommand = theApp.m_pBrowserCommandMap->GetFEResource(url);
|
|
WFE_GetOwnerFrame(this)->PostMessage(WM_COMMAND, MAKEWPARAM(nCommand, nCommand), 0);
|
|
}
|
|
else if (!HT_IsContainer(m_Node) && !HT_IsSeparator(m_Node))
|
|
{
|
|
CAbstractCX * pCX = FEU_GetLastActiveFrameContext();
|
|
ASSERT(pCX != NULL);
|
|
if (pCX != NULL)
|
|
{
|
|
if (!HT_Launch(m_Node, pCX->GetContext()))
|
|
pCX->NormalGetUrl((LPTSTR)HT_GetNodeURL(m_Node));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CRDFToolbarButton::FillInOleDataSource(COleDataSource *pDataSource)
|
|
{
|
|
CLIPFORMAT cfHTNode = (CLIPFORMAT)RegisterClipboardFormat(NETSCAPE_HTNODE_FORMAT);
|
|
HANDLE hString = GlobalAlloc(GMEM_MOVEABLE|GMEM_DDESHARE,1);
|
|
pDataSource->CacheGlobalData(cfHTNode, hString);
|
|
|
|
// Need to "select" the button in the view. Hack.
|
|
HT_View theView = HT_GetView(m_Node);
|
|
HT_SetSelectedView(HT_GetPane(theView), theView); // Make sure this view is selected in the pane.
|
|
HT_SetSelection(m_Node);
|
|
|
|
// Now the view is sufficient
|
|
RDFGLOBAL_BeginDrag(pDataSource, HT_GetView(m_Node));
|
|
|
|
}
|
|
|
|
BOOKMARKITEM CRDFToolbarButton::GetBookmarkItem(void)
|
|
{
|
|
return m_bookmark;
|
|
}
|
|
|
|
void CRDFToolbarButton::SetBookmarkItem(BOOKMARKITEM bookmark)
|
|
{
|
|
/* m_bookmark = bookmark;
|
|
CString strText(bookmark.szText);
|
|
CToolbarButton::SetText(strText);
|
|
*/
|
|
}
|
|
|
|
void CRDFToolbarButton::SetNode(HT_Resource pNode, BOOL bAddRef)
|
|
{
|
|
if (pNode == NULL)
|
|
return;
|
|
|
|
m_Node = pNode;
|
|
|
|
// if it's a header and we haven't already loaded the bitmaps, load the bitmaps
|
|
if(HT_IsContainer(pNode) && bAddRef)
|
|
{
|
|
HDC hDC = ::GetDC(m_hWnd);
|
|
|
|
HINSTANCE hInstance = AfxGetResourceHandle();
|
|
WFE_InitializeUIPalette(hDC);
|
|
HPALETTE hPalette = WFE_GetUIPalette(GetParentFrame());
|
|
if(nBitmapRefCount == 0)
|
|
{
|
|
m_hbmpBM = WFE_LoadTransparentBitmap(hInstance, hDC, sysInfo.m_clrMenu, RGB(255, 0, 255),
|
|
hPalette, IDB_BOOKMARK_ITEM);
|
|
m_hbmpBMSelected = WFE_LoadTransparentBitmap(hInstance, hDC,sysInfo.m_clrHighlight,
|
|
RGB(255, 0, 255), hPalette, IDB_BOOKMARK_ITEM);
|
|
m_hbmpFolder = WFE_LoadTransparentBitmap(hInstance, hDC,sysInfo.m_clrMenu, RGB(255, 0, 255),
|
|
hPalette, IDB_BOOKMARK_FOLDER2);
|
|
|
|
m_hbmpSelectedFolder = WFE_LoadTransparentBitmap(hInstance, hDC, sysInfo.m_clrHighlight, RGB(255, 0, 255),
|
|
hPalette, IDB_BOOKMARK_FOLDER2);
|
|
|
|
m_hbmpFolderOpen = WFE_LoadTransparentBitmap(hInstance, hDC, sysInfo.m_clrHighlight,
|
|
RGB(255, 0, 255), hPalette, IDB_BOOKMARK_FOLDER_OPEN);
|
|
}
|
|
nBitmapRefCount++;
|
|
|
|
::ReleaseDC(m_hWnd, hDC);
|
|
}
|
|
|
|
}
|
|
|
|
void CRDFToolbarButton::EditTextChanged(char *pText)
|
|
{
|
|
if (pText)
|
|
{
|
|
HT_SetNodeName(m_Node, pText);
|
|
SetText(pText);
|
|
SetToolTipText(pText);
|
|
delete []pText;
|
|
}
|
|
RemoveTextEdit();
|
|
|
|
if (foundOnRDFToolbar())
|
|
((CRDFToolbar*)GetParent())->LayoutButtons(-1);
|
|
}
|
|
|
|
void CRDFToolbarButton::DrawPicturesAndTextMode(HDC hDC, CRect rect)
|
|
{
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* theToolbar = (CRDFToolbar*)GetParent();
|
|
void* data;
|
|
HT_GetTemplateData(HT_TopNode(theToolbar->GetHTView()), gNavCenter->toolbarBitmapPosition, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
CString position((char*)data);
|
|
if (position == "top")
|
|
{
|
|
DrawBitmapOnTop(hDC, rect);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
DrawBitmapOnSide(hDC, rect);
|
|
}
|
|
|
|
void CRDFToolbarButton::DrawButtonText(HDC hDC, CRect rcTxt, CSize sizeTxt, CString strTxt)
|
|
{
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
hasCustomTextColor = TRUE;
|
|
switch (m_eState)
|
|
{
|
|
case eDISABLED:
|
|
customTextColor = ((CRDFToolbar*)GetParent())->GetDisabledColor();
|
|
if (customTextColor == -1)
|
|
{
|
|
hasCustomTextColor = FALSE;
|
|
}
|
|
break;
|
|
case eBUTTON_UP:
|
|
customTextColor = ((CRDFToolbar*)GetParent())->GetRolloverColor();
|
|
break;
|
|
case eBUTTON_DOWN:
|
|
customTextColor = ((CRDFToolbar*)GetParent())->GetPressedColor();
|
|
break;
|
|
case eNORMAL:
|
|
customTextColor = ((CRDFToolbar*)GetParent())->GetForegroundColor();
|
|
break;
|
|
}
|
|
|
|
if (m_bDepressed)
|
|
customTextColor = ((CRDFToolbar*)GetParent())->GetPressedColor();
|
|
}
|
|
|
|
CToolbarButton::DrawButtonText(hDC, rcTxt, sizeTxt, strTxt);
|
|
}
|
|
|
|
CSize CRDFToolbarButton::GetButtonSizeFromChars(CString s, int c)
|
|
{
|
|
if(m_nToolbarStyle == TB_PICTURESANDTEXT)
|
|
{
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* theToolbar = (CRDFToolbar*)GetParent();
|
|
void* data;
|
|
HT_GetTemplateData(HT_TopNode(theToolbar->GetHTView()), gNavCenter->toolbarBitmapPosition, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
CString position((char*)data);
|
|
if (position == "top")
|
|
{
|
|
return GetBitmapOnTopSize(s, c);
|
|
}
|
|
}
|
|
}
|
|
return GetBitmapOnSideSize(s, c);
|
|
}
|
|
else if(m_nToolbarStyle == TB_PICTURES)
|
|
return GetBitmapOnlySize();
|
|
else // m_nToolbarStyle == TB_TEXT
|
|
return GetTextOnlySize(s, c);
|
|
}
|
|
|
|
void CRDFToolbarButton::GetPicturesAndTextModeTextRect(CRect &rect)
|
|
{
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* theToolbar = (CRDFToolbar*)GetParent();
|
|
void* data;
|
|
HT_GetTemplateData(HT_TopNode(theToolbar->GetHTView()), gNavCenter->toolbarBitmapPosition, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
CString position((char*)data);
|
|
if (position == "top")
|
|
{
|
|
GetBitmapOnTopTextRect(rect);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
GetBitmapOnSideTextRect(rect);
|
|
}
|
|
|
|
void CRDFToolbarButton::GetPicturesModeTextRect(CRect &rect)
|
|
{
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* theToolbar = (CRDFToolbar*)GetParent();
|
|
void* data;
|
|
HT_GetTemplateData(HT_TopNode(theToolbar->GetHTView()), gNavCenter->toolbarBitmapPosition, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
CString position((char*)data);
|
|
if (position == "top")
|
|
{
|
|
GetBitmapOnTopTextRect(rect);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
GetBitmapOnSideTextRect(rect);
|
|
}
|
|
|
|
|
|
BOOL CRDFToolbarButton::CreateRightMouseMenu(void)
|
|
{
|
|
if (m_bShouldShowRMMenu)
|
|
{
|
|
m_MenuCommandMap.Clear();
|
|
HT_View theView = HT_GetView(m_Node);
|
|
HT_SetSelectedView(HT_GetPane(theView), theView); // Make sure this view is selected in the pane.
|
|
HT_SetSelection(m_Node); // Make sure the node is the selection in the view.
|
|
|
|
HT_Cursor theCursor = HT_NewContextualMenuCursor(theView, PR_FALSE, PR_FALSE);
|
|
if (theCursor != NULL)
|
|
{
|
|
// We have a cursor. Attempt to iterate
|
|
HT_MenuCmd theCommand;
|
|
while (HT_NextContextMenuItem(theCursor, &theCommand))
|
|
{
|
|
char* menuName = HT_GetMenuCmdName(theCommand);
|
|
if (theCommand == HT_CMD_SEPARATOR)
|
|
m_menu.AppendMenu(MF_SEPARATOR);
|
|
else
|
|
{
|
|
// Add the command to our command map
|
|
CRDFMenuCommand* rdfCommand = new CRDFMenuCommand(menuName, theCommand);
|
|
int index = m_MenuCommandMap.AddCommand(rdfCommand);
|
|
m_menu.AppendMenu(MF_ENABLED, index+FIRST_HT_MENU_ID, menuName);
|
|
}
|
|
}
|
|
HT_DeleteCursor(theCursor);
|
|
}
|
|
}
|
|
return m_bShouldShowRMMenu;
|
|
}
|
|
|
|
CWnd* CRDFToolbarButton::GetMenuParent(void)
|
|
{
|
|
return this;
|
|
}
|
|
|
|
BOOL CRDFToolbarButton::OnCommand(UINT wParam, LONG lParam)
|
|
{
|
|
BOOL bRtn = TRUE;
|
|
|
|
if (wParam >= FIRST_HT_MENU_ID && wParam <= LAST_HT_MENU_ID)
|
|
{
|
|
// A selection was made from the context menu.
|
|
// Use the menu map to get the HT command value
|
|
CRDFMenuCommand* theCommand = (CRDFMenuCommand*)(m_MenuCommandMap.GetCommand((int)wParam-FIRST_HT_MENU_ID));
|
|
if (theCommand)
|
|
{
|
|
HT_MenuCmd htCommand = theCommand->GetHTCommand();
|
|
HT_DoMenuCmd(HT_GetPane(GetHTView()), htCommand);
|
|
}
|
|
|
|
return bRtn;
|
|
}
|
|
|
|
if(wParam == ID_HOTLIST_VIEW || wParam == ID_HOTLIST_ADDCURRENTTOHOTLIST)
|
|
{
|
|
GetParentFrame()->SendMessage(WM_COMMAND, wParam, lParam);
|
|
}
|
|
else if(wParam == ID_DELETE_BUTTON)
|
|
{
|
|
HT_RemoveChild(HT_GetParent(m_Node), m_Node);
|
|
}
|
|
else if(wParam == ID_BUTTON_PROPERTIES)
|
|
{
|
|
// PROPERTIES
|
|
}
|
|
else if(wParam == ID_RENAME_BUTTON)
|
|
{
|
|
AddTextEdit();
|
|
SetTextEditText(m_pButtonText);
|
|
}
|
|
// Only interested in commands from bookmark quick file menu
|
|
else if ((wParam >= FIRST_BOOKMARK_MENU_ID))
|
|
{
|
|
void * pBookmark = NULL;
|
|
m_BMMenuMap.Lookup(wParam, pBookmark);
|
|
if (pBookmark != NULL)
|
|
{
|
|
HT_Resource theNode = (HT_Resource)pBookmark;
|
|
CAbstractCX *pCX = FEU_GetLastActiveFrameContext();
|
|
ASSERT(pCX != NULL);
|
|
if (pCX != NULL)
|
|
{
|
|
if (!HT_Launch(theNode, pCX->GetContext()))
|
|
pCX->NormalGetUrl((LPTSTR)HT_GetNodeURL(theNode));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
bRtn = CRDFToolbarButtonBase::OnCommand(wParam, lParam);
|
|
}
|
|
|
|
return(bRtn);
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
// CRDFToolbarButton Messages
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
BEGIN_MESSAGE_MAP(CRDFToolbarButton, CRDFToolbarButtonBase)
|
|
//{{AFX_MSG_MAP(CRDFToolbarButton)
|
|
ON_MESSAGE(NSDRAGMENUOPEN, OnDragMenuOpen)
|
|
ON_MESSAGE(DM_FILLINMENU, OnFillInMenu)
|
|
ON_MESSAGE(DT_DROPOCCURRED, OnDropMenuDropOccurred)
|
|
ON_MESSAGE(DT_DRAGGINGOCCURRED, OnDropMenuDraggingOccurred)
|
|
ON_MESSAGE(DM_MENUCLOSED, OnDropMenuClosed)
|
|
ON_WM_SYSCOLORCHANGE()
|
|
ON_WM_PAINT()
|
|
ON_WM_MOUSEACTIVATE()
|
|
//}}AFX_MSG_MAP
|
|
|
|
END_MESSAGE_MAP()
|
|
|
|
|
|
int CRDFToolbarButton::OnMouseActivate( CWnd* pDesktopWnd, UINT nHitTest, UINT message )
|
|
{
|
|
m_bButtonDown = TRUE;
|
|
return MA_ACTIVATE;
|
|
}
|
|
|
|
|
|
void CRDFToolbarButton::OnPaint()
|
|
{
|
|
CRect updateRect;
|
|
|
|
GetUpdateRect(&updateRect);
|
|
|
|
CPaintDC dcPaint(this); // device context for painting
|
|
CRect rcClient;
|
|
GetClientRect(&rcClient);
|
|
|
|
HDC hSrcDC = dcPaint.m_hDC;
|
|
|
|
HPALETTE hPalette;
|
|
HPALETTE hOldPalette;
|
|
CFrameWnd* pParent = WFE_GetOwnerFrame(this);
|
|
hPalette = WFE_GetUIPalette(pParent);
|
|
hOldPalette= ::SelectPalette(hSrcDC, hPalette, FALSE);
|
|
HDC hMemDC = ::CreateCompatibleDC(hSrcDC);
|
|
|
|
if(hMemDC)
|
|
{
|
|
HPALETTE hOldMemPalette = ::SelectPalette(hMemDC, hPalette, FALSE);
|
|
|
|
HBITMAP hbmMem = CreateCompatibleBitmap(hSrcDC,
|
|
rcClient.Width(),
|
|
rcClient.Height());
|
|
|
|
//
|
|
// Select the bitmap into the off-screen DC.
|
|
//
|
|
HBITMAP hbmOld = (HBITMAP)::SelectObject(hMemDC, hbmMem);
|
|
|
|
// if we are not enabled then must make sure that button state is normal
|
|
if(!m_bEnabled)
|
|
{
|
|
m_eState = eNORMAL;
|
|
}
|
|
|
|
CRect innerRect = rcClient;
|
|
|
|
innerRect.InflateRect(-BORDERSIZE, -BORDERSIZE);
|
|
|
|
int oldMode = ::SetBkMode(hMemDC, TRANSPARENT);
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)GetParent();
|
|
if (pToolbar->GetBackgroundImage() == NULL ||
|
|
(pToolbar->GetBackgroundImage() != NULL &&
|
|
!pToolbar->GetBackgroundImage()->FrameSuccessfullyLoaded()))
|
|
{
|
|
// Fill with our background color.
|
|
HBRUSH hRegBrush = (HBRUSH) ::CreateSolidBrush(pToolbar->GetBackgroundColor());
|
|
::FillRect(hMemDC, rcClient, hRegBrush);
|
|
}
|
|
else
|
|
{
|
|
// There is a background. Let's do a tile on the rect corresponding to our position
|
|
// in the parent toolbar.
|
|
CWnd* pGrandParent = pToolbar->GetParent();
|
|
CRect offsetRect(rcClient);
|
|
MapWindowPoints(pGrandParent, &offsetRect);
|
|
|
|
// Now we want to fill the given rectangle.
|
|
PaintBackground(hMemDC, rcClient, pToolbar->GetBackgroundImage(), offsetRect.left, offsetRect.top);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
::FillRect(hMemDC, rcClient, sysInfo.m_hbrBtnFace);
|
|
}
|
|
|
|
if(m_nToolbarStyle == TB_PICTURESANDTEXT)
|
|
{
|
|
DrawPicturesAndTextMode(hMemDC, innerRect);
|
|
}
|
|
else if(m_nToolbarStyle == TB_PICTURES)
|
|
{
|
|
DrawPicturesMode(hMemDC, innerRect);
|
|
}
|
|
else
|
|
{
|
|
DrawTextMode(hMemDC, innerRect);
|
|
}
|
|
|
|
// Now, draw 3d visual button effects, depending on our state
|
|
switch (m_eState)
|
|
{
|
|
case eBUTTON_UP:
|
|
{
|
|
if( m_nChecked == 0 )
|
|
DrawUpButton(hMemDC, rcClient);
|
|
else
|
|
DrawDownButton(hMemDC, rcClient);
|
|
}
|
|
break;
|
|
|
|
case eBUTTON_CHECKED:
|
|
{
|
|
// A checked button but NOT mousing over - no black border
|
|
DrawCheckedButton(hMemDC, rcClient);
|
|
}
|
|
break;
|
|
|
|
case eBUTTON_DOWN:
|
|
{
|
|
DrawDownButton(hMemDC, rcClient);
|
|
}
|
|
break;
|
|
|
|
case eDISABLED:
|
|
{
|
|
if(m_nChecked == 2)
|
|
DrawCheckedButton(hMemDC, rcClient);
|
|
|
|
}
|
|
break;
|
|
|
|
case eNORMAL:
|
|
{
|
|
if (m_bDepressed)
|
|
DrawDownButton(hMemDC, rcClient); // Looks like it's locked down.
|
|
}
|
|
}
|
|
|
|
::SetBkMode(hMemDC, oldMode);
|
|
|
|
::BitBlt(hSrcDC, 0, 0, rcClient.Width(), rcClient.Height(), hMemDC, 0, 0,
|
|
SRCCOPY);
|
|
|
|
::SelectPalette(hMemDC, hOldMemPalette, FALSE);
|
|
::SelectPalette(hSrcDC, hOldPalette, FALSE);
|
|
|
|
::SelectObject(hMemDC, hbmOld);
|
|
::DeleteObject(hbmMem);
|
|
|
|
::DeleteDC(hMemDC);
|
|
}
|
|
}
|
|
|
|
LRESULT CRDFToolbarButton::OnDragMenuOpen(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
// Set our drop menu's user data.
|
|
if (m_pDropMenu)
|
|
m_pDropMenu->SetUserData(m_Node);
|
|
|
|
if(m_Node == NULL || !HT_IsContainer(m_Node))
|
|
return 1;
|
|
m_uCurBMID = FIRST_BOOKMARK_MENU_ID;
|
|
|
|
m_pCachedDropMenu = (CDropMenu *)lParam; // Set our drop menu
|
|
PRBool isOpen;
|
|
|
|
HT_Resource theNode = (HT_Resource)m_pCachedDropMenu->GetUserData();
|
|
HT_GetOpenState(theNode, &isOpen);
|
|
if (isOpen)
|
|
FillInMenu(theNode);
|
|
else HT_SetOpenState(theNode, (PRBool)TRUE);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
LRESULT CRDFToolbarButton::OnFillInMenu(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
// Set our drop menu's user data.
|
|
if (m_pDropMenu)
|
|
m_pDropMenu->SetUserData(m_Node);
|
|
|
|
m_pCachedDropMenu = (CDropMenu *)lParam; // Set our drop menu
|
|
PRBool isOpen;
|
|
HT_Resource theNode = (HT_Resource)m_pCachedDropMenu->GetUserData();
|
|
HT_GetOpenState(theNode, &isOpen);
|
|
if (isOpen)
|
|
FillInMenu(theNode);
|
|
else HT_SetOpenState(theNode, (PRBool)TRUE);
|
|
|
|
return 1;
|
|
}
|
|
|
|
LRESULT CRDFToolbarButton::OnDropMenuDraggingOccurred(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
CDropMenuDragData* pData = (CDropMenuDragData*)lParam;
|
|
UINT nCommand = pData->m_nCommand;
|
|
MenuSelectionType eSelType = pData->eSelType;
|
|
|
|
void * pBookmarkInsertAfter = NULL;
|
|
m_BMMenuMap.Lookup(nCommand, pBookmarkInsertAfter);
|
|
HT_Resource theNode = (HT_Resource)pBookmarkInsertAfter;
|
|
|
|
int dragFraction = 2;
|
|
if (eSelType == eON)
|
|
dragFraction = 2;
|
|
else if (eSelType == eBELOW)
|
|
dragFraction = 3;
|
|
else if (eSelType == eABOVE)
|
|
dragFraction = 1;
|
|
|
|
// Next we get the result.
|
|
DROPEFFECT answer = RDFGLOBAL_TranslateDropAction(theNode, pData->m_pDataSource, dragFraction);
|
|
|
|
// Place the result into our data structure for the droptarget to use.
|
|
pData->m_DropEffect = answer;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
LRESULT CRDFToolbarButton::OnDropMenuDropOccurred(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
CDropMenuDragData* pData = (CDropMenuDragData*)lParam;
|
|
UINT nCommand = pData->m_nCommand;
|
|
MenuSelectionType eSelType = pData->eSelType;
|
|
|
|
void * pBookmarkInsertAfter = NULL;
|
|
m_BMMenuMap.Lookup(nCommand, pBookmarkInsertAfter);
|
|
HT_Resource theNode = (HT_Resource)pBookmarkInsertAfter;
|
|
|
|
int dragFraction = 2;
|
|
if (eSelType == eON)
|
|
dragFraction = 2;
|
|
else if (eSelType == eBELOW)
|
|
dragFraction = 3;
|
|
else if (eSelType == eABOVE)
|
|
dragFraction = 1;
|
|
|
|
RDFGLOBAL_PerformDrop(pData->m_pDataSource, theNode, dragFraction);
|
|
|
|
delete pData; // Clean this structure up.
|
|
|
|
return 1;
|
|
}
|
|
|
|
LRESULT CRDFToolbarButton::OnDropMenuClosed(WPARAM wParam, LPARAM lParam)
|
|
{
|
|
int nCount;
|
|
if(m_pDropMenu != NULL)
|
|
{
|
|
HT_Resource theNode = (HT_Resource)m_pDropMenu->GetUserData();
|
|
nCount = m_pDropMenu->GetMenuItemCount();
|
|
|
|
// clean out the menu
|
|
for(int i = nCount - 1; i >= 0; i--)
|
|
{
|
|
m_pDropMenu->DeleteMenu(i, MF_BYPOSITION);
|
|
}
|
|
m_pDropMenu->DestroyDropMenu();
|
|
delete m_pDropMenu;
|
|
m_pDropMenu = NULL;
|
|
|
|
if (theNode != NULL)
|
|
HT_SetOpenState(theNode, (PRBool)FALSE);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
void CRDFToolbarButton::OnSysColorChange( )
|
|
{
|
|
if(nBitmapRefCount > 0)
|
|
{
|
|
VERIFY(::DeleteObject(m_hbmpBM));
|
|
VERIFY(::DeleteObject(m_hbmpBMSelected));
|
|
VERIFY(::DeleteObject(m_hbmpFolder));
|
|
VERIFY(::DeleteObject(m_hbmpSelectedFolder));
|
|
VERIFY(::DeleteObject(m_hbmpFolderOpen));
|
|
|
|
HINSTANCE hInstance = AfxGetResourceHandle();
|
|
HDC hDC = ::GetDC(m_hWnd);
|
|
WFE_InitializeUIPalette(hDC);
|
|
HPALETTE hPalette = WFE_GetUIPalette(GetParentFrame());
|
|
|
|
m_hbmpBM = WFE_LoadTransparentBitmap(hInstance, hDC, sysInfo.m_clrMenu, RGB(255, 0, 255),
|
|
hPalette, IDB_BOOKMARK_ITEM);
|
|
m_hbmpBMSelected = WFE_LoadTransparentBitmap(hInstance, hDC,sysInfo.m_clrHighlight,
|
|
RGB(255, 0, 255), hPalette, IDB_BOOKMARK_ITEM);
|
|
m_hbmpFolder = WFE_LoadTransparentBitmap(hInstance, hDC,sysInfo.m_clrMenu, RGB(255, 0, 255),
|
|
hPalette, IDB_BOOKMARK_FOLDER2);
|
|
|
|
m_hbmpSelectedFolder = WFE_LoadTransparentBitmap(hInstance, hDC,sysInfo.m_clrHighlight, RGB(255, 0, 255),
|
|
hPalette, IDB_BOOKMARK_FOLDER2);
|
|
|
|
m_hbmpFolderOpen = WFE_LoadTransparentBitmap(hInstance, hDC, sysInfo.m_clrHighlight,
|
|
RGB(255, 0, 255), hPalette, IDB_BOOKMARK_FOLDER_OPEN);
|
|
|
|
::ReleaseDC(m_hWnd, hDC);
|
|
}
|
|
|
|
if(m_bBitmapFromParent)
|
|
{
|
|
HDC hDC = ::GetDC(m_hWnd);
|
|
|
|
HPALETTE hPalette = WFE_GetUIPalette(GetParentFrame());
|
|
|
|
HBITMAP hBitmap = WFE_LookupLoadAndEnterBitmap(hDC, m_nBitmapID, TRUE, hPalette,
|
|
sysInfo.m_clrBtnFace, RGB(255, 0, 255));
|
|
::ReleaseDC(m_hWnd, hDC);
|
|
SetBitmap(hBitmap, TRUE);
|
|
}
|
|
CToolbarButton::OnSysColorChange();
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// CRDFToolbarButton Helpers
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
void CRDFToolbarButton::FillInMenu(HT_Resource theNode)
|
|
{
|
|
// Determine our position
|
|
/* CWnd* frame = GetTopLevelFrame();
|
|
CPoint point(0, GetRequiredButtonSize().cy);
|
|
MapWindowPoints(frame, &point, 1);
|
|
*/
|
|
|
|
CRDFToolbarHolder* pHolder = (CRDFToolbarHolder*)(GetParent()->GetParent()->GetParent());
|
|
CRDFToolbarButton* pOldButton = pHolder->GetCurrentButton();
|
|
if (pOldButton != NULL)
|
|
{
|
|
pOldButton->GetTreeView()->DeleteNavCenter();
|
|
}
|
|
pHolder->SetCurrentButton(NULL);
|
|
|
|
if (!m_bDepressed && pOldButton != this)
|
|
{
|
|
CPoint point = RequestMenuPlacement();
|
|
m_pTreeView = CNSNavFrame::CreateFramedRDFViewFromResource(NULL, point.x, point.y, 300, 500, m_Node);
|
|
CRDFOutliner* pOutliner = (CRDFOutliner*)(m_pTreeView->GetContentView()->GetOutlinerParent()->GetOutliner());
|
|
if (pOutliner->IsPopup() || XP_IsNavCenterDocked(m_pTreeView->GetHTPane()))
|
|
{
|
|
SetDepressed(TRUE);
|
|
m_pTreeView->SetRDFButton(this);
|
|
pHolder->SetCurrentButton(this);
|
|
}
|
|
else
|
|
{
|
|
m_pTreeView = NULL;
|
|
}
|
|
}
|
|
|
|
/*
|
|
m_pCachedDropMenu->CreateOverflowMenuItem(ID_HOTLIST_VIEW, CString(szLoadString(IDS_MORE_BOOKMARKS)), NULL, NULL );
|
|
|
|
HT_Cursor theCursor = HT_NewCursor(theNode);
|
|
HT_Resource theItem = NULL;
|
|
while (theItem = HT_GetNextItem(theCursor))
|
|
{
|
|
IconType nIconType = DetermineIconType(theItem, FALSE);
|
|
void* pCustomIcon = NULL;
|
|
if (nIconType == LOCAL_FILE)
|
|
pCustomIcon = FetchLocalFileIcon(theItem);
|
|
else if (nIconType == ARBITRARY_URL)
|
|
pCustomIcon = FetchCustomIcon(theItem, m_pCachedDropMenu, FALSE);
|
|
|
|
HT_SetNodeFEData(theItem, this);
|
|
|
|
if (HT_IsContainer(theItem))
|
|
{
|
|
CDropMenu *pSubMenu = new CDropMenu;
|
|
pSubMenu->SetUserData(theItem);
|
|
|
|
CString csAmpersandString = FEU_EscapeAmpersand(HT_GetNodeName(theItem));
|
|
|
|
m_pCachedDropMenu->AppendMenu(MF_POPUP, m_uCurBMID, pSubMenu, FALSE,
|
|
csAmpersandString, TRUE, m_hbmpFolder, m_hbmpFolderOpen, pCustomIcon, nIconType);
|
|
|
|
m_BMMenuMap.SetAt(m_uCurBMID, theItem);
|
|
m_uCurBMID++;
|
|
}
|
|
else if (!HT_IsSeparator(theItem))
|
|
{
|
|
CString csAmpersandString = FEU_EscapeAmpersand(HT_GetNodeName(theItem));
|
|
m_pCachedDropMenu->AppendMenu(MF_STRING, m_uCurBMID, csAmpersandString, TRUE,
|
|
m_hbmpBM, m_hbmpBMSelected, pCustomIcon, nIconType);
|
|
m_BMMenuMap.SetAt(m_uCurBMID, theItem);
|
|
m_uCurBMID++;
|
|
}
|
|
else
|
|
{
|
|
m_pCachedDropMenu->AppendMenu(MF_SEPARATOR, 0, TRUE, NULL, NULL);
|
|
}
|
|
}
|
|
|
|
HT_DeleteCursor(theCursor);
|
|
m_pCachedDropMenu = NULL;
|
|
*/
|
|
}
|
|
|
|
UINT CRDFToolbarButton::GetBitmapID(void)
|
|
{
|
|
if (m_Node)
|
|
{
|
|
if (strncmp(HT_GetNodeURL(m_Node), "command:", 8) == 0)
|
|
{
|
|
// We're an internal command.
|
|
return IDB_PICTURES;
|
|
}
|
|
else if (HT_IsContainer(m_Node))
|
|
return IDB_BUTTON_FOLDER;
|
|
else return IDB_USERBTN;
|
|
}
|
|
|
|
return IDB_USERBTN;
|
|
}
|
|
|
|
UINT CRDFToolbarButton::GetBitmapIndex(void)
|
|
{
|
|
if (m_Node)
|
|
{
|
|
if (strncmp(HT_GetNodeURL(m_Node), "command:", 8) == 0)
|
|
{
|
|
// We're an internal command. Access the command bitmap strip.
|
|
return theApp.m_pCommandToolbarIndices->GetFEResource(HT_GetNodeURL(m_Node));
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void CRDFToolbarButton::SetTextWithoutResize(CString text)
|
|
{
|
|
if(m_pButtonText != NULL)
|
|
{
|
|
XP_FREE(m_pButtonText);
|
|
}
|
|
|
|
m_pButtonText = (LPTSTR)XP_ALLOC(text.GetLength() +1);
|
|
XP_STRCPY(m_pButtonText, text);
|
|
}
|
|
|
|
BOOL CRDFToolbarButton::NeedsUpdate()
|
|
{
|
|
// Only internal commands need updating via the OnUpdateCmdUI handler.
|
|
// All other buttons are enabled/disabled using the appropriate HT calls.
|
|
if (m_Node && strncmp(HT_GetNodeURL(m_Node), "command:", 8) == 0)
|
|
{
|
|
m_nCommand = theApp.m_pBrowserCommandMap->GetFEResource(HT_GetNodeURL(m_Node));
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
void CRDFToolbarButton::LoadComplete(HT_Resource r)
|
|
{
|
|
Invalidate();
|
|
}
|
|
|
|
void CRDFToolbarButton::DrawCustomIcon(HDC hDC, int x, int y)
|
|
{
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)GetParent();
|
|
|
|
CRDFImage* pCustomImage = DrawArbitraryURL(m_Node, x, y, m_bitmapSize.cx, m_bitmapSize.cy, hDC,
|
|
pToolbar->GetBackgroundColor(),
|
|
this, UseLargeIcons());
|
|
|
|
if (!pCustomImage->FrameSuccessfullyLoaded())
|
|
return;
|
|
|
|
// Adjust the toolbar button's width and height.
|
|
long width = pCustomImage->bmpInfo->bmiHeader.biWidth;
|
|
long height = pCustomImage->bmpInfo->bmiHeader.biHeight;
|
|
|
|
if (width > m_bitmapSize.cx || height > m_bitmapSize.cy)
|
|
{
|
|
SetBitmapSize(CSize(width, height));
|
|
|
|
CSize buttonSize = GetMinimalButtonSize(); // Only care about height.
|
|
|
|
// Grow the toolbar if necessary.
|
|
if (buttonSize.cy > pToolbar->GetRowHeight())
|
|
{
|
|
pToolbar->SetRowHeight(buttonSize.cy);
|
|
pToolbar->LayoutButtons(-1);
|
|
GetParentFrame()->RecalcLayout();
|
|
}
|
|
else
|
|
{
|
|
// We're too small. Need to adjust our bitmap height.
|
|
int diff = pToolbar->GetRowHeight() - buttonSize.cy;
|
|
SetBitmapSize(CSize(width, height + diff));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void CRDFToolbarButton::DrawLocalIcon(HDC hDC, int x, int y)
|
|
{
|
|
if (m_Node)
|
|
DrawLocalFileIcon(m_Node, x, y, hDC);
|
|
}
|
|
|
|
void CRDFToolbarButton::DrawButtonBitmap(HDC hDC, CRect rcImg)
|
|
{
|
|
BTN_STATE eState = m_eState;
|
|
|
|
if(m_eState == eBUTTON_CHECKED)
|
|
// A checked button has same bitmap as the normal state with no mouse-over
|
|
eState = eNORMAL;
|
|
else if(m_eState == eBUTTON_UP && m_nChecked == 2)
|
|
// if we are in the mouse over mode, but indeterminate we want our bitmap to have a disabled look
|
|
eState = eDISABLED;
|
|
|
|
UpdateIconInfo();
|
|
|
|
// Create a scratch DC and select our bitmap into it.
|
|
HDC pBmpDC = ::CreateCompatibleDC(hDC);
|
|
HPALETTE hPalette = WFE_GetUIPalette(GetParentFrame());
|
|
CBitmap BmpImg;
|
|
CPoint ptDst;
|
|
|
|
HBITMAP hOldBmp;
|
|
HPALETTE hOldPal;
|
|
if (m_hBmpImg != NULL)
|
|
{
|
|
HINSTANCE hInst = AfxGetResourceHandle();
|
|
HBITMAP hBmpImg;
|
|
hBmpImg = m_hBmpImg;
|
|
hOldBmp = (HBITMAP)::SelectObject(pBmpDC, hBmpImg);
|
|
hOldPal = ::SelectPalette(pBmpDC, WFE_GetUIPalette(NULL), TRUE);
|
|
::RealizePalette(pBmpDC);
|
|
}
|
|
|
|
// Get the image dimensions
|
|
|
|
int realBitmapHeight;
|
|
if(m_nIconType == LOCAL_FILE)
|
|
{
|
|
realBitmapHeight = 16;
|
|
m_bitmapSize.cx = 16;
|
|
}
|
|
else if (m_nIconType == ARBITRARY_URL)
|
|
{
|
|
realBitmapHeight = m_bitmapSize.cy;
|
|
}
|
|
else
|
|
{
|
|
if (m_nBitmapID == IDB_PICTURES)
|
|
realBitmapHeight = 21; // Height of command buttons
|
|
else realBitmapHeight = 17; // Height of personal toolbar button bitmaps.
|
|
}
|
|
|
|
// Center the image within the button
|
|
ptDst.x = (rcImg.Width() >= m_bitmapSize.cx) ?
|
|
rcImg.left + (((rcImg.Width() - m_bitmapSize.cx) + 1) / 2) : rcImg.left;
|
|
|
|
ptDst.y = (rcImg.Height() >= realBitmapHeight) ?
|
|
rcImg.top + (((rcImg.Height() - realBitmapHeight) + 1) / 2) : rcImg.top;
|
|
|
|
// If we're in the checked state, shift the image one pixel
|
|
if (m_eState == eBUTTON_CHECKED || (m_eState == eBUTTON_UP && m_nChecked == 1))
|
|
{
|
|
ptDst.x += 1;
|
|
ptDst.y += 1;
|
|
}
|
|
|
|
// Call the handy transparent blit function to paint the bitmap over
|
|
// whatever colors exist.
|
|
|
|
CPoint bitmapStart;
|
|
|
|
if(m_bIsResourceID)
|
|
bitmapStart = CPoint(m_nBitmapIndex * m_bitmapSize.cx, m_bEnabled ? realBitmapHeight * eState : realBitmapHeight);
|
|
|
|
if(m_bIsResourceID)
|
|
{
|
|
|
|
if(m_nIconType == LOCAL_FILE)
|
|
{
|
|
DrawLocalIcon(hDC, ptDst.x, ptDst.y);
|
|
}
|
|
else if (m_nIconType == ARBITRARY_URL)
|
|
{
|
|
DrawCustomIcon(hDC, ptDst.x, ptDst.y);
|
|
}
|
|
else
|
|
FEU_TransBlt( hDC, ptDst.x, ptDst.y, m_bitmapSize.cx, realBitmapHeight,
|
|
pBmpDC, bitmapStart.x, bitmapStart.y ,WFE_GetUIPalette(NULL),
|
|
GetSysColor(COLOR_BTNFACE));
|
|
//::BitBlt(hDC, ptDst.x, ptDst.y, m_bitmapSize.cx, realBitmapHeight,
|
|
// pBmpDC, bitmapStart.x, bitmapStart.y, SRCCOPY);
|
|
}
|
|
|
|
if (m_hBmpImg != NULL)
|
|
{
|
|
// Cleanup
|
|
::SelectObject(pBmpDC, hOldBmp);
|
|
::SelectPalette(pBmpDC, hOldPal, TRUE);
|
|
::DeleteDC(pBmpDC);
|
|
}
|
|
}
|
|
|
|
HFONT CRDFToolbarButton::GetFont(HDC hDC)
|
|
{
|
|
return CToolbarButton::GetFont(hDC);
|
|
}
|
|
|
|
int CRDFToolbarButton::DrawText(HDC hDC, LPCSTR lpString, int nCount, LPRECT lpRect, UINT uFormat)
|
|
{
|
|
// XP_ASSERT(IsUTF8Text(lpString, nCount));
|
|
// In RDF, everything is UTF8
|
|
return CIntlWin::DrawText(CS_UTF8, hDC, (char*)lpString, nCount, lpRect, uFormat );
|
|
}
|
|
|
|
BOOL CRDFToolbarButton::GetTextExtentPoint32(HDC hDC, LPCSTR lpString, int nCount, LPSIZE lpSize)
|
|
{
|
|
// XP_ASSERT(IsUTF8Text(lpString, nCount));
|
|
// In RDF, everything is UTF8
|
|
return CIntlWin::GetTextExtentPoint(CS_UTF8, hDC, (char*)lpString, nCount, lpSize);
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// Class CRDFSeparatorButton
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
BEGIN_MESSAGE_MAP(CRDFSeparatorButton, CRDFToolbarButton)
|
|
//{{AFX_MSG_MAP(CRDFSeparatorButton)
|
|
//}}AFX_MSG_MAP
|
|
|
|
END_MESSAGE_MAP()
|
|
|
|
CSize CRDFSeparatorButton::GetButtonSizeFromChars(CString s, int c)
|
|
{
|
|
return CSize(8, 8);
|
|
}
|
|
|
|
void CRDFSeparatorButton::DrawButtonBitmap(HDC hDC, CRect rcImg)
|
|
{
|
|
CDC* pDC = CDC::FromHandle(hDC);
|
|
|
|
CRect clientRect;
|
|
GetClientRect(&clientRect);
|
|
|
|
COLORREF shadowColor = ::GetSysColor(COLOR_3DSHADOW);
|
|
COLORREF highlightColor = ::GetSysColor(COLOR_3DLIGHT);
|
|
|
|
if (foundOnRDFToolbar())
|
|
{
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)GetParent();
|
|
shadowColor = pToolbar->GetShadowColor();
|
|
highlightColor = pToolbar->GetHighlightColor();
|
|
}
|
|
|
|
HPEN hShadowPen = (HPEN)::CreatePen(PS_SOLID, 1, shadowColor);
|
|
HPEN hHighlightPen = (HPEN)::CreatePen(PS_SOLID, 1, highlightColor);
|
|
|
|
HPEN hOldPen = (HPEN)(pDC->SelectObject(hShadowPen));
|
|
pDC->MoveTo(3, 3);
|
|
pDC->LineTo(3, clientRect.bottom - 2);
|
|
|
|
(HPEN)(pDC->SelectObject(hHighlightPen));
|
|
pDC->MoveTo(4, 3);
|
|
pDC->LineTo(4, clientRect.bottom - 2);
|
|
|
|
pDC->SelectObject(hOldPen);
|
|
|
|
VERIFY(::DeleteObject(hShadowPen));
|
|
VERIFY(::DeleteObject(hHighlightPen));
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// Class CRDFToolbarDropTarget
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
void CRDFToolbarDropTarget::Toolbar(CRDFToolbar *pToolbar)
|
|
{
|
|
m_pToolbar = pToolbar;
|
|
}
|
|
|
|
DROPEFFECT CRDFToolbarDropTarget::OnDragEnter(CWnd * pWnd, COleDataObject * pDataObject,
|
|
DWORD dwKeyState, CPoint point)
|
|
{
|
|
return OnDragOver(pWnd, pDataObject, dwKeyState, point);
|
|
}
|
|
|
|
DROPEFFECT CRDFToolbarDropTarget::OnDragOver(CWnd * pWnd, COleDataObject * pDataObject,
|
|
DWORD dwKeyState, CPoint point )
|
|
{
|
|
int nStartX = 0;
|
|
|
|
RECT buttonRect;
|
|
|
|
int currentRow = 0;
|
|
|
|
CRDFToolbarButton* pButton = NULL;
|
|
|
|
for(int i = 0; i < m_pToolbar->GetNumButtons(); i++)
|
|
{
|
|
pButton = (CRDFToolbarButton*)(m_pToolbar->GetNthButton(i));
|
|
pButton->GetClientRect(&buttonRect);
|
|
pButton->MapWindowPoints(m_pToolbar, &buttonRect);
|
|
|
|
nStartX += (buttonRect.right - buttonRect.left) + SPACE_BETWEEN_BUTTONS;
|
|
if (currentRow != pButton->GetRow())
|
|
{
|
|
currentRow++;
|
|
nStartX = LEFT_TOOLBAR_MARGIN + (buttonRect.right - buttonRect.left) + SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
|
|
if(point.x < nStartX && (point.y >= buttonRect.top && point.y <= buttonRect.bottom))
|
|
break;
|
|
|
|
pButton = NULL;
|
|
}
|
|
|
|
HT_Resource theNode = pButton ? pButton->GetNode() : HT_TopNode(m_pToolbar->GetHTView());
|
|
|
|
m_pToolbar->SetDragButton(pButton);
|
|
|
|
if (pButton == NULL)
|
|
m_pToolbar->SetDragFraction(2);
|
|
else
|
|
{
|
|
// Do the whole computation of drag fraction. Cache our drag fraction and button.
|
|
CRect rect;
|
|
pButton->GetClientRect(&rect);
|
|
if (HT_IsContainer(pButton->GetNode()))
|
|
{
|
|
if (point.x <= rect.Width()/3)
|
|
m_pToolbar->SetDragFraction(1);
|
|
else if (point.x <= 2*(rect.Width()/3))
|
|
m_pToolbar->SetDragFraction(2);
|
|
else m_pToolbar->SetDragFraction(3);
|
|
}
|
|
else if (point.x <= rect.Width()/2)
|
|
m_pToolbar->SetDragFraction(1);
|
|
else m_pToolbar->SetDragFraction(2);
|
|
}
|
|
|
|
return RDFGLOBAL_TranslateDropAction(theNode, pDataObject, m_pToolbar->GetDragFraction());
|
|
}
|
|
|
|
BOOL CRDFToolbarDropTarget::OnDrop(CWnd * pWnd, COleDataObject * pDataObject,
|
|
DROPEFFECT dropEffect, CPoint point)
|
|
{
|
|
HT_Resource theNode = HT_TopNode(m_pToolbar->GetHTView());
|
|
if (m_pToolbar->GetDragButton())
|
|
theNode = m_pToolbar->GetDragButton()->GetNode();
|
|
|
|
RDFGLOBAL_PerformDrop(pDataObject, theNode, m_pToolbar->GetDragFraction());
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
// End CRDFToolbarDropTarget implementation
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// Class CRDFToolbar
|
|
///////////////////////////////////////////////////////////////////////////
|
|
#define LINKTOOLBARHEIGHT 21
|
|
#define COMMANDTOOLBARHEIGHT 42
|
|
#define SPACE_BETWEEN_ROWS 2
|
|
|
|
// The Event Handler for HT notifications on the toolbars
|
|
static void toolbarNotifyProcedure (HT_Notification ns, HT_Resource n, HT_Event whatHappened,
|
|
void *token, uint32 tokenType)
|
|
{
|
|
static int toolbarIDCounter = 0;
|
|
|
|
CRDFToolbarHolder* theToolbarHolder = (CRDFToolbarHolder*)ns->data;
|
|
if (theToolbarHolder == NULL)
|
|
return;
|
|
|
|
HT_View theView = HT_GetView(n);
|
|
|
|
// The pane has to handle some events. These will go here.
|
|
if (whatHappened == HT_EVENT_VIEW_SELECTED)
|
|
{
|
|
|
|
}
|
|
|
|
if (theView == NULL)
|
|
return;
|
|
|
|
if (whatHappened == HT_EVENT_VIEW_ADDED)
|
|
{
|
|
CRDFToolbar* theNewToolbar = CRDFToolbar::CreateUserToolbar(theView, theToolbarHolder->GetCachedParentWindow());
|
|
CButtonToolbarWindow *pWindow = new CButtonToolbarWindow(theNewToolbar,
|
|
theApp.m_pToolbarStyle, 43, 27, eSMALL_HTAB);
|
|
|
|
theToolbarHolder->AddNewWindow(ID_PERSONAL_TOOLBAR+toolbarIDCounter, pWindow, toolbarIDCounter, 43, 27, 1,
|
|
HT_GetNodeName(HT_TopNode(theNewToolbar->GetHTView())),theApp.m_pToolbarStyle, FALSE);
|
|
toolbarIDCounter++;
|
|
theToolbarHolder->GetCachedParentWindow()->RecalcLayout();
|
|
}
|
|
else if (whatHappened == HT_EVENT_VIEW_DELETED)
|
|
{
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)HT_GetViewFEData(theView);
|
|
pToolbar->SetHTView(NULL);
|
|
delete pToolbar;
|
|
HT_SetViewFEData(theView, NULL);
|
|
}
|
|
else if (whatHappened == HT_EVENT_NODE_VPROP_CHANGED && HT_TopNode(theView) == n)
|
|
{
|
|
}
|
|
else if (whatHappened == HT_EVENT_NODE_EDIT && HT_TopNode(theView) == n)
|
|
{
|
|
// Edit being performed on a selector bar item. (STILL TO DO)
|
|
}
|
|
else if (whatHappened == HT_EVENT_VIEW_WORKSPACE_REFRESH)
|
|
{
|
|
}
|
|
// If the pane doesn't handle the event, then a view does.
|
|
else
|
|
{
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)HT_GetViewFEData(theView);
|
|
if (pToolbar != NULL)
|
|
pToolbar->HandleEvent(ns, n, whatHappened);
|
|
}
|
|
}
|
|
|
|
void CRDFToolbar::HandleEvent(HT_Notification ns, HT_Resource n, HT_Event whatHappened)
|
|
{
|
|
HT_View theView = m_ToolbarView;
|
|
if (theView != NULL)
|
|
{
|
|
if (whatHappened == HT_EVENT_NODE_OPENCLOSE_CHANGED)
|
|
{
|
|
PRBool openState;
|
|
HT_GetOpenState(n, &openState);
|
|
if (openState)
|
|
{
|
|
if (n == HT_TopNode(theView))
|
|
{
|
|
// Initial population of the toolbar. We should only receive this event once.
|
|
FillInToolbar();
|
|
}
|
|
else
|
|
{
|
|
// Toolbar button menu. Populate it.
|
|
CRDFToolbarButton* theButton = (CRDFToolbarButton*)(HT_GetNodeFEData(n));
|
|
theButton->FillInMenu(n);
|
|
}
|
|
}
|
|
}
|
|
else if (whatHappened == HT_EVENT_VIEW_REFRESH)
|
|
{
|
|
LayoutButtons(-1);
|
|
}
|
|
else if (HT_TopNode(theView) == HT_GetParent(n))
|
|
{
|
|
// Aside from the opening/closing, we only respond to events that occurred on the top node of
|
|
// the view or the immediate children (the buttons on the toolbar).
|
|
if (whatHappened == HT_EVENT_NODE_DELETED_DATA ||
|
|
whatHappened == HT_EVENT_NODE_DELETED_NODATA)
|
|
{
|
|
// Delete the button
|
|
if (HT_EVENT_NODE_DELETED_DATA == whatHappened)
|
|
{
|
|
// Destroy the toolbar button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)HT_GetNodeFEData(n);
|
|
if (m_hWnd)
|
|
RemoveButton(pButton, FALSE);
|
|
else
|
|
DecrementButtonCount();
|
|
delete pButton;
|
|
|
|
}
|
|
}
|
|
else if (whatHappened == HT_EVENT_NODE_ADDED)
|
|
{
|
|
AddHTButton(n);
|
|
LayoutButtons(-1);
|
|
}
|
|
else if (whatHappened == HT_EVENT_NODE_EDIT)
|
|
{
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)HT_GetNodeFEData(n);
|
|
pButton->AddTextEdit();
|
|
}
|
|
else if (whatHappened == HT_EVENT_NODE_VPROP_CHANGED)
|
|
{
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)HT_GetNodeFEData(n);
|
|
if (pButton->m_hWnd)
|
|
LayoutButtons(-1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
CRDFToolbar::CRDFToolbar(HT_View htView, int nMaxButtons, int nToolbarStyle, int nPicturesAndTextHeight,
|
|
int nPicturesHeight, int nTextHeight)
|
|
: CNSToolbar2(nMaxButtons, nToolbarStyle, nPicturesAndTextHeight, nPicturesHeight, nTextHeight)
|
|
{
|
|
// Set our view and point HT at us.
|
|
m_pBackgroundImage = NULL;
|
|
m_ToolbarView = htView;
|
|
HT_SetViewFEData(htView, this);
|
|
|
|
m_nNumberOfRows = 1;
|
|
m_nRowHeight = LINKTOOLBARHEIGHT;
|
|
void* data;
|
|
HT_GetTemplateData(HT_TopNode(GetHTView()), gNavCenter->toolbarBitmapPosition, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
CString position((char*)data);
|
|
if (position == "top")
|
|
m_nRowHeight = COMMANDTOOLBARHEIGHT;
|
|
}
|
|
}
|
|
|
|
CRDFToolbar* CRDFToolbar::CreateUserToolbar(HT_View theView, CWnd* pParent)
|
|
{
|
|
// fetch the RDF toolbar style property
|
|
char *data;
|
|
HT_GetTemplateData(HT_TopNode(theView), gNavCenter->toolbarDisplayMode, HT_COLUMN_STRING,
|
|
(void **)&data);
|
|
int style = StyleFromHTDescriptor(data);
|
|
if (style < 0)
|
|
style = theApp.m_pToolbarStyle;
|
|
|
|
CRDFToolbar* pToolbar = new CRDFToolbar(theView, MAX_TOOLBAR_BUTTONS, style, 43, 27, 27);
|
|
|
|
if (pToolbar->Create(pParent))
|
|
{
|
|
// Top node is already open. Fill it in.
|
|
PRBool openState;
|
|
HT_Resource topNode = HT_TopNode(theView);
|
|
HT_GetOpenState(topNode, &openState);
|
|
if (openState)
|
|
pToolbar->FillInToolbar();
|
|
else HT_SetOpenState(topNode, PR_TRUE); // Let the callback kick in.
|
|
}
|
|
|
|
return pToolbar;
|
|
}
|
|
|
|
CRDFToolbar::~CRDFToolbar()
|
|
{
|
|
m_ToolbarView = NULL;
|
|
}
|
|
|
|
int CRDFToolbar::Create(CWnd *pParent)
|
|
{
|
|
|
|
int result = CNSToolbar2::Create(pParent);
|
|
|
|
if(!result)
|
|
return FALSE;
|
|
|
|
m_DropTarget.Register(this);
|
|
m_DropTarget.Toolbar(this);
|
|
|
|
DragAcceptFiles(FALSE);
|
|
|
|
HT_Resource topNode = HT_TopNode(GetHTView());
|
|
BOOL fixedSize = FALSE;
|
|
|
|
void* data;
|
|
|
|
HT_GetTemplateData(topNode, gNavCenter->toolbarButtonsFixedSize, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
CString answer((char*)data);
|
|
if (answer.GetLength() > 0 && (answer.GetAt(0) == 'y' || answer.GetAt(0) == 'Y'))
|
|
fixedSize = TRUE;
|
|
}
|
|
SetButtonsSameWidth(fixedSize);
|
|
|
|
|
|
return result;
|
|
}
|
|
|
|
void CRDFToolbar::FillInToolbar()
|
|
{
|
|
if (!m_ToolbarView)
|
|
return;
|
|
|
|
HT_Resource top = HT_TopNode(m_ToolbarView);
|
|
HT_Cursor cursor = HT_NewCursor(top);
|
|
if (cursor == NULL)
|
|
return;
|
|
|
|
HT_Resource item = NULL;
|
|
while (item = HT_GetNextItem(cursor))
|
|
AddHTButton(item);
|
|
|
|
HT_DeleteCursor(cursor);
|
|
|
|
LayoutButtons(-1);
|
|
}
|
|
|
|
int CRDFToolbar::GetDisplayMode(void)
|
|
{
|
|
char *data;
|
|
int style;
|
|
HT_Resource top = HT_TopNode(GetHTView());
|
|
HT_GetTemplateData(top, gNavCenter->toolbarDisplayMode, HT_COLUMN_STRING, (void **)&data);
|
|
style = data ? StyleFromHTDescriptor(data) : -1;
|
|
return style >= 0 ? style : theApp.m_pToolbarStyle;
|
|
}
|
|
|
|
void CRDFToolbar::AddHTButton(HT_Resource item)
|
|
{
|
|
BOOKMARKITEM bookmark;
|
|
XP_STRCPY(bookmark.szText, HT_GetNodeName(item));
|
|
XP_STRCPY(bookmark.szAnchor, HT_GetNodeURL(item));
|
|
CString csAmpersandString = FEU_EscapeAmpersand(CString(bookmark.szText));
|
|
CString tooltipText(bookmark.szText); // Default is to use the name for the tooltip
|
|
CString statusBarText(bookmark.szAnchor); // and the URL for the status bar text.
|
|
|
|
// Fetch the button's tooltip and status bar text.
|
|
void* data;
|
|
HT_GetTemplateData(item, gNavCenter->buttonTooltipText, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
tooltipText = (char*)data;
|
|
HT_GetTemplateData(item, gNavCenter->buttonStatusbarText, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
statusBarText = (char*)data;
|
|
|
|
CRDFToolbarButton* pButton = NULL;
|
|
if (HT_IsURLBar(item))
|
|
pButton = new CURLBarButton;
|
|
else if (HT_IsSeparator(item))
|
|
{
|
|
pButton = new CRDFSeparatorButton;
|
|
tooltipText = "Separator";
|
|
statusBarText = "Separator";
|
|
}
|
|
else pButton = new CRDFToolbarButton;
|
|
|
|
pButton->Create(this, GetDisplayMode(), CSize(60,42), CSize(85, 25), csAmpersandString,
|
|
tooltipText, statusBarText, CSize(23,17),
|
|
m_nMaxToolbarButtonChars, m_nMinToolbarButtonChars, bookmark,
|
|
item, (HT_IsContainer(item) ? TB_HAS_DRAGABLE_MENU | TB_HAS_IMMEDIATE_MENU : 0));
|
|
|
|
if(HT_IsContainer(item))
|
|
{
|
|
CRDFToolbarButtonDropTarget *pDropTarget = new CRDFToolbarButtonDropTarget;
|
|
pButton->SetDropTarget(pDropTarget);
|
|
}
|
|
|
|
HT_SetNodeFEData(item, pButton);
|
|
CSize buttonSize = pButton->GetMinimalButtonSize(); // Only care about height.
|
|
|
|
if (buttonSize.cy > m_nRowHeight)
|
|
{
|
|
m_nRowHeight = buttonSize.cy;
|
|
GetParentFrame()->RecalcLayout();
|
|
}
|
|
else if (buttonSize.cy < m_nRowHeight)
|
|
{
|
|
CSize size = pButton->GetBitmapSize();
|
|
pButton->SetBitmapSize(CSize(size.cx, size.cy + (m_nRowHeight - buttonSize.cy)));
|
|
}
|
|
|
|
m_pButtonArray[m_nNumButtons] = pButton;
|
|
|
|
m_nNumButtons++;
|
|
|
|
if(CheckMaxButtonSizeChanged(pButton, TRUE) && !HT_IsURLBar(item) && !HT_IsSeparator(item))
|
|
{
|
|
ChangeButtonSizes();
|
|
}
|
|
else
|
|
{
|
|
CSize size = pButton->GetRequiredButtonSize();
|
|
int nWidth = size.cx;
|
|
|
|
if (m_bButtonsSameWidth && !HT_IsURLBar(item) && !HT_IsSeparator(item))
|
|
nWidth = m_nMaxButtonWidth;
|
|
|
|
//make sure it's the size of the largest button so far
|
|
pButton->SetButtonSize(CSize(nWidth, m_nMaxButtonHeight));
|
|
}
|
|
|
|
// Update the button if it's a command.
|
|
if (pButton->NeedsUpdate())
|
|
{
|
|
pButton->OnUpdateCmdUI(GetTopLevelFrame(), FALSE);
|
|
}
|
|
}
|
|
|
|
void CRDFToolbar::ChangeButtonSizes(void)
|
|
{
|
|
int nWidth;
|
|
|
|
HT_Cursor cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
HT_Resource item;
|
|
while (item = HT_GetNextItem(cursor))
|
|
{
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!HT_IsSeparator(item) && !HT_IsURLBar(item) && pButton)
|
|
{
|
|
if(!m_bButtonsSameWidth)
|
|
{
|
|
CSize size = pButton->GetRequiredButtonSize();
|
|
nWidth = size.cx;
|
|
}
|
|
else
|
|
nWidth = m_nMaxButtonWidth;
|
|
|
|
pButton->SetButtonSize(CSize(nWidth, m_nMaxButtonHeight));
|
|
}
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
}
|
|
|
|
/* RDF toolbars get their style info from RDF. App preference settings
|
|
make their way into RDF elsewhere. */
|
|
void CRDFToolbar::SetToolbarStyle(int nToolbarStyle)
|
|
{
|
|
CNSToolbar2::SetToolbarStyle(GetDisplayMode());
|
|
}
|
|
|
|
int CRDFToolbar::GetHeight(void)
|
|
{
|
|
return m_nNumberOfRows * (m_nRowHeight + SPACE_BETWEEN_ROWS) + SPACE_BETWEEN_ROWS;
|
|
}
|
|
|
|
void CRDFToolbar::SetMinimumRows(int rowWidth)
|
|
{
|
|
int rowCount = 1;
|
|
int totalLine = 0;
|
|
int rowSpace = rowWidth - RIGHT_TOOLBAR_MARGIN - LEFT_TOOLBAR_MARGIN;
|
|
|
|
if (rowSpace <= 0)
|
|
{
|
|
SetRows(rowCount);
|
|
return;
|
|
}
|
|
|
|
HT_Cursor cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
HT_Resource item;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
CSize s = pButton->GetMinimalButtonSize();
|
|
int tempTotal = totalLine + s.cx;
|
|
if (tempTotal > rowSpace)
|
|
{
|
|
rowCount++;
|
|
totalLine = s.cx;
|
|
}
|
|
else totalLine = tempTotal;
|
|
|
|
totalLine += SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
SetRows(rowCount);
|
|
}
|
|
|
|
void CRDFToolbar::ComputeLayoutInfo(CRDFToolbarButton* pButton, int numChars, int rowWidth,
|
|
int& usedSpace)
|
|
{
|
|
CString originalText = HT_GetNodeName(pButton->GetNode());
|
|
|
|
int currCount = originalText.GetLength();
|
|
|
|
// Start off at the maximal string
|
|
CString strTxt = originalText;
|
|
|
|
if (currCount > numChars)
|
|
{
|
|
strTxt = originalText.Left(numChars-3) + "...";
|
|
}
|
|
|
|
pButton->SetTextWithoutResize(strTxt);
|
|
|
|
if (!m_bButtonsSameWidth || HT_IsURLBar(pButton->GetNode()) || HT_IsSeparator(pButton->GetNode()))
|
|
pButton->SetButtonSize(pButton->GetButtonSizeFromChars(strTxt, numChars));
|
|
|
|
// Determine how much additional padding we'll use to fill out a row if this button doesn't fit.
|
|
int rowUsage = usedSpace % rowWidth;
|
|
if (rowUsage == 0)
|
|
rowUsage = rowWidth;
|
|
int additionalPadding = rowWidth - rowUsage;
|
|
|
|
int tempTotal = rowUsage + pButton->GetButtonSize().cx;
|
|
|
|
// The button doesn't fit. Flesh out this row and start a new one.
|
|
if (tempTotal > rowWidth)
|
|
usedSpace += additionalPadding;
|
|
|
|
// Add this button to the row.
|
|
usedSpace += pButton->GetButtonSize().cx + SPACE_BETWEEN_BUTTONS;
|
|
|
|
// Set this button's row information, so it knows which row it is currently residing on.
|
|
int currentRow = usedSpace/rowWidth;
|
|
if (usedSpace % rowWidth == 0)
|
|
currentRow--;
|
|
|
|
pButton->SetRow(currentRow);
|
|
}
|
|
|
|
void CRDFToolbar::LayoutButtons(int nIndex)
|
|
{
|
|
// Make sure buttons exist for every item.
|
|
HT_Cursor cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
HT_Resource item;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
{
|
|
AddHTButton(item);
|
|
pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
}
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
int width = m_nWidth;
|
|
|
|
if (width <= 0)
|
|
{
|
|
CRect rect;
|
|
GetParentFrame()->GetClientRect(&rect);
|
|
width = rect.Width();
|
|
}
|
|
|
|
int numButtonsAtMin = 0;
|
|
int numButtonsAtMax = 0;
|
|
int idealSpace = 0;
|
|
|
|
// First quickly determine what the minimum # of rows we consume is. This is our allowed space.
|
|
int oldRows = GetRows();
|
|
|
|
int rowWidth = width-RIGHT_TOOLBAR_MARGIN-LEFT_TOOLBAR_MARGIN;
|
|
|
|
if (rowWidth <= 0 && m_nWidth > 0)
|
|
rowWidth = m_nWidth - RIGHT_TOOLBAR_MARGIN - LEFT_TOOLBAR_MARGIN;
|
|
|
|
SetMinimumRows(width);
|
|
|
|
int newRows = GetRows();
|
|
|
|
int allowedSpace = rowWidth * GetRows(); // Toolbar width * numRows
|
|
int usedSpace = 0;
|
|
|
|
int numChars = 0; // Start off trying to fit the whole thing on the toolbar.
|
|
int minChars = 0;
|
|
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
if (numChars == 0)
|
|
{
|
|
numChars = pButton->GetMaxTextCharacters();
|
|
minChars = pButton->GetMinTextCharacters();
|
|
}
|
|
|
|
// See how much this num chars takes up
|
|
ComputeLayoutInfo(pButton, numChars, rowWidth, usedSpace);
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
while (usedSpace > allowedSpace && numChars > minChars)
|
|
{
|
|
usedSpace = 0;
|
|
numChars--;
|
|
// Let's see what we can fit.
|
|
HT_Cursor cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
HT_Resource item;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton) // Separator
|
|
continue;
|
|
|
|
// See how much this num chars takes up
|
|
ComputeLayoutInfo(pButton, numChars, rowWidth, usedSpace);
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
}
|
|
|
|
int nStartX = LEFT_TOOLBAR_MARGIN;
|
|
int nStartY = SPACE_BETWEEN_ROWS;
|
|
|
|
int row = 0;
|
|
|
|
// Now we have to handle springs
|
|
// Initialize our spring counter (we're using this to track the number of springs
|
|
// on each toolbar row).
|
|
int* springCounter = new int[newRows];
|
|
int* rowPadding = new int[newRows];
|
|
for (int init=0; init < newRows; init++) { springCounter[init] = 0; }
|
|
for (int init2=0; init2 < newRows; init2++) { rowPadding[init2] = 0; }
|
|
|
|
// Count the springs on each row, as well as the padding on each row.
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
if (pButton->IsSpring())
|
|
{
|
|
springCounter[row]++;
|
|
}
|
|
|
|
CSize buttonSize = pButton->GetButtonSize(); // The size we must be
|
|
|
|
int tempTotal = nStartX + buttonSize.cx;
|
|
if (tempTotal > (width - RIGHT_TOOLBAR_MARGIN))
|
|
{
|
|
// Compute row padding
|
|
int rowPad = width - RIGHT_TOOLBAR_MARGIN - nStartX;
|
|
|
|
// Store the row padding information
|
|
rowPadding[row] = rowPad;
|
|
|
|
// Move to the next row.
|
|
nStartX = LEFT_TOOLBAR_MARGIN;
|
|
nStartY += m_nRowHeight + SPACE_BETWEEN_ROWS;
|
|
row++;
|
|
}
|
|
|
|
nStartX += buttonSize.cx + SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
rowPadding[row] = width - RIGHT_TOOLBAR_MARGIN - nStartX;
|
|
HT_DeleteCursor(cursor);
|
|
|
|
// Now we know the number of springs on each row, and we know how much available padding
|
|
// we have on each row. Expand the springs' sizes so that all space on rows with springs
|
|
// is consumed.
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
CSize buttonSize = pButton->GetButtonSize(); // The size we must be
|
|
int row = pButton->GetRow();
|
|
|
|
if (pButton->IsSpring())
|
|
{
|
|
// Expand the spring
|
|
int numSprings = springCounter[row];
|
|
int rowPad = rowPadding[row];
|
|
int addPadding = rowPad;
|
|
if (numSprings > 1)
|
|
{
|
|
springCounter[row]--;
|
|
addPadding = rowPad / numSprings;
|
|
rowPadding[row] -= addPadding;
|
|
}
|
|
|
|
pButton->SetButtonSize(CSize(buttonSize.cx + addPadding, buttonSize.cy));
|
|
|
|
}
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
// Clean up. Delete our temporary arrays.
|
|
delete []springCounter;
|
|
delete []rowPadding;
|
|
|
|
// That's it. lay them out with this number of characters.
|
|
nStartX = LEFT_TOOLBAR_MARGIN;
|
|
nStartY = SPACE_BETWEEN_ROWS;
|
|
row = 0;
|
|
CSize buttonSize;
|
|
CString strTxt;
|
|
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
buttonSize = pButton->GetButtonSize(); // The size we must be
|
|
|
|
int tempTotal = nStartX + buttonSize.cx;
|
|
if (tempTotal > (width - RIGHT_TOOLBAR_MARGIN))
|
|
{
|
|
nStartX = LEFT_TOOLBAR_MARGIN;
|
|
nStartY += m_nRowHeight + SPACE_BETWEEN_ROWS;
|
|
}
|
|
|
|
if (buttonSize.cy < m_nRowHeight)
|
|
{
|
|
CSize size = pButton->GetBitmapSize();
|
|
pButton->SetBitmapSize(CSize(size.cx, size.cy + (m_nRowHeight - buttonSize.cy)));
|
|
}
|
|
|
|
pButton->MoveWindow(nStartX, nStartY,
|
|
buttonSize.cx, m_nRowHeight);
|
|
|
|
nStartX += buttonSize.cx + SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
m_nWidth = width;
|
|
|
|
if (oldRows != newRows)
|
|
GetParentFrame()->RecalcLayout();
|
|
}
|
|
|
|
void CRDFToolbar::WidthChanged(int animWidth)
|
|
{
|
|
CRect rect;
|
|
|
|
GetParentFrame()->GetClientRect(&rect);
|
|
int width = rect.Width() - animWidth;
|
|
|
|
int numButtonsAtMin = 0;
|
|
int numButtonsAtMax = 0;
|
|
int idealSpace = 0;
|
|
|
|
// First quickly determine what the minimum # of rows we consume is. This is our allowed space.
|
|
int oldRows = GetRows();
|
|
|
|
int rowWidth = width-RIGHT_TOOLBAR_MARGIN-LEFT_TOOLBAR_MARGIN;
|
|
|
|
if (rowWidth <= 0 && m_nWidth > 0)
|
|
rowWidth = m_nWidth - RIGHT_TOOLBAR_MARGIN - LEFT_TOOLBAR_MARGIN;
|
|
|
|
SetMinimumRows(width);
|
|
|
|
int newRows = GetRows();
|
|
|
|
int allowedSpace = rowWidth * GetRows(); // Toolbar width * numRows
|
|
int usedSpace = 0;
|
|
|
|
int numChars = 0; // Start off trying to fit the whole thing on the toolbar.
|
|
int minChars = 0;
|
|
HT_Cursor cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
HT_Resource item;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton) // Separator
|
|
continue;
|
|
|
|
if (numChars == 0)
|
|
{
|
|
numChars = pButton->GetMaxTextCharacters();
|
|
minChars = pButton->GetMinTextCharacters();
|
|
}
|
|
|
|
// See how much this num chars takes up
|
|
ComputeLayoutInfo(pButton, numChars, rowWidth, usedSpace);
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
while (usedSpace > allowedSpace && numChars > minChars)
|
|
{
|
|
usedSpace = 0;
|
|
numChars--;
|
|
// Let's see what we can fit.
|
|
HT_Cursor cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
HT_Resource item;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
// See how much this num chars takes up
|
|
ComputeLayoutInfo(pButton, numChars, rowWidth, usedSpace);
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
}
|
|
|
|
// That's it. lay them out with this number of characters.
|
|
|
|
int nStartX = LEFT_TOOLBAR_MARGIN;
|
|
int nStartY = SPACE_BETWEEN_ROWS;
|
|
|
|
int row = 0;
|
|
|
|
// Now we have to handle springs
|
|
// Initialize our spring counter (we're using this to track the number of springs
|
|
// on each toolbar row).
|
|
int* springCounter = new int[newRows];
|
|
int* rowPadding = new int[newRows];
|
|
for (int init=0; init < newRows; init++) { springCounter[init] = 0; }
|
|
for (int init2=0; init2 < newRows; init2++) { rowPadding[init2] = 0; }
|
|
|
|
// Count the springs on each row, as well as the padding on each row.
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
if (pButton->IsSpring())
|
|
{
|
|
springCounter[row]++;
|
|
}
|
|
|
|
CSize buttonSize = pButton->GetButtonSize(); // The size we must be
|
|
|
|
int tempTotal = nStartX + buttonSize.cx;
|
|
if (tempTotal > (width - RIGHT_TOOLBAR_MARGIN))
|
|
{
|
|
// Compute row padding
|
|
int rowPad = width - RIGHT_TOOLBAR_MARGIN - nStartX;
|
|
|
|
// Store the row padding information
|
|
rowPadding[row] = rowPad;
|
|
|
|
// Move to the next row.
|
|
nStartX = LEFT_TOOLBAR_MARGIN;
|
|
nStartY += m_nRowHeight + SPACE_BETWEEN_ROWS;
|
|
row++;
|
|
}
|
|
|
|
nStartX += buttonSize.cx + SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
rowPadding[row] = width - RIGHT_TOOLBAR_MARGIN - nStartX;
|
|
HT_DeleteCursor(cursor);
|
|
|
|
// Now we know the number of springs on each row, and we know how much available padding
|
|
// we have on each row. Expand the springs' sizes so that all space on rows with springs
|
|
// is consumed.
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
CSize buttonSize = pButton->GetButtonSize(); // The size we must be
|
|
int row = pButton->GetRow();
|
|
|
|
if (pButton->IsSpring())
|
|
{
|
|
// Expand the spring
|
|
int numSprings = springCounter[row];
|
|
int rowPad = rowPadding[row];
|
|
int addPadding = rowPad;
|
|
if (numSprings > 1)
|
|
{
|
|
springCounter[row]--;
|
|
addPadding = rowPad / numSprings;
|
|
rowPadding[row] -= addPadding;
|
|
}
|
|
|
|
pButton->SetButtonSize(CSize(buttonSize.cx + addPadding, buttonSize.cy));
|
|
|
|
}
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
// Clean up. Delete our temporary arrays.
|
|
delete []springCounter;
|
|
delete []rowPadding;
|
|
|
|
nStartX = LEFT_TOOLBAR_MARGIN;
|
|
nStartY = SPACE_BETWEEN_ROWS;
|
|
row = 0;
|
|
|
|
CSize buttonSize;
|
|
CString strTxt;
|
|
|
|
cursor = HT_NewCursor(HT_TopNode(m_ToolbarView));
|
|
if (!cursor)
|
|
return;
|
|
|
|
while ((item = HT_GetNextItem(cursor)))
|
|
{
|
|
// Get the current button
|
|
CRDFToolbarButton* pButton = (CRDFToolbarButton*)(HT_GetNodeFEData(item));
|
|
if (!pButton)
|
|
continue;
|
|
|
|
buttonSize = pButton->GetButtonSize(); // The size we must be
|
|
|
|
int tempTotal = nStartX + buttonSize.cx;
|
|
if (tempTotal > (width - RIGHT_TOOLBAR_MARGIN))
|
|
{
|
|
nStartX = LEFT_TOOLBAR_MARGIN;
|
|
nStartY += m_nRowHeight + SPACE_BETWEEN_ROWS;
|
|
}
|
|
|
|
if (buttonSize.cy < m_nRowHeight)
|
|
{
|
|
CSize size = pButton->GetBitmapSize();
|
|
pButton->SetBitmapSize(CSize(size.cx, size.cy + (m_nRowHeight - buttonSize.cy)));
|
|
}
|
|
|
|
pButton->MoveWindow(nStartX, nStartY,
|
|
buttonSize.cx, m_nRowHeight);
|
|
|
|
nStartX += buttonSize.cx + SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
HT_DeleteCursor(cursor);
|
|
|
|
|
|
m_nWidth = width;
|
|
|
|
if (oldRows != newRows)
|
|
GetParentFrame()->RecalcLayout();
|
|
}
|
|
|
|
int CRDFToolbar::GetRowWidth()
|
|
{
|
|
return m_nWidth - RIGHT_TOOLBAR_MARGIN - LEFT_TOOLBAR_MARGIN - SPACE_BETWEEN_BUTTONS;
|
|
}
|
|
|
|
void CRDFToolbar::ComputeColorsForSeparators()
|
|
{
|
|
// background color
|
|
HT_Resource top = HT_TopNode(GetHTView());
|
|
void* data;
|
|
COLORREF backgroundColor = GetSysColor(COLOR_BTNFACE);
|
|
COLORREF shadowColor = backgroundColor;
|
|
COLORREF highlightColor = backgroundColor;
|
|
|
|
HT_GetTemplateData(top, gNavCenter->viewBGColor, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
WFE_ParseColor((char*)data, &backgroundColor);
|
|
}
|
|
Compute3DColors(backgroundColor, highlightColor, shadowColor);
|
|
SetBackgroundColor(backgroundColor);
|
|
SetHighlightColor(highlightColor);
|
|
SetShadowColor(shadowColor);
|
|
}
|
|
|
|
void CRDFToolbar::OnPaint(void)
|
|
{
|
|
CRect rcClient, updateRect, buttonRect, intersectRect;
|
|
|
|
GetClientRect(&rcClient);
|
|
GetUpdateRect(&updateRect);
|
|
|
|
CPaintDC dcPaint(this);
|
|
|
|
// background color
|
|
HT_Resource top = HT_TopNode(GetHTView());
|
|
void* data;
|
|
COLORREF backgroundColor = GetSysColor(COLOR_BTNFACE);
|
|
COLORREF shadowColor = backgroundColor;
|
|
COLORREF highlightColor = backgroundColor;
|
|
|
|
HT_GetTemplateData(top, gNavCenter->viewBGColor, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
WFE_ParseColor((char*)data, &backgroundColor);
|
|
}
|
|
Compute3DColors(backgroundColor, highlightColor, shadowColor);
|
|
|
|
if (m_BackgroundColor != backgroundColor)
|
|
{
|
|
// Ensure proper filling in of parent space with the background color.
|
|
SetBackgroundColor(backgroundColor);
|
|
GetParent()->Invalidate();
|
|
}
|
|
|
|
SetShadowColor(shadowColor);
|
|
SetHighlightColor(highlightColor);
|
|
|
|
// Foreground color
|
|
COLORREF foregroundColor = GetSysColor(COLOR_BTNTEXT);
|
|
HT_GetTemplateData(top, gNavCenter->viewFGColor, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
WFE_ParseColor((char*)data, &foregroundColor);
|
|
}
|
|
SetForegroundColor(foregroundColor);
|
|
|
|
// Rollover color
|
|
COLORREF rolloverColor = RGB(0, 0, 255);
|
|
HT_GetTemplateData(top, gNavCenter->viewRolloverColor, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
WFE_ParseColor((char*)data, &rolloverColor);
|
|
}
|
|
SetRolloverColor(rolloverColor);
|
|
|
|
// Pressed color
|
|
COLORREF pressedColor = RGB(0, 0, 128);
|
|
HT_GetTemplateData(top, gNavCenter->viewPressedColor, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
WFE_ParseColor((char*)data, &pressedColor);
|
|
}
|
|
SetPressedColor(pressedColor);
|
|
|
|
// Disabled color
|
|
COLORREF disabledColor = -1;
|
|
HT_GetTemplateData(top, gNavCenter->viewDisabledColor, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
{
|
|
WFE_ParseColor((char*)data, &disabledColor);
|
|
}
|
|
SetDisabledColor(disabledColor);
|
|
|
|
// Background image URL
|
|
CString backgroundImageURL = "";
|
|
HT_GetTemplateData(top, gNavCenter->viewBGURL, HT_COLUMN_STRING, &data);
|
|
if (data)
|
|
backgroundImageURL = (char*)data;
|
|
SetBackgroundImage(NULL); // Clear out the BG image.
|
|
|
|
HBRUSH hRegBrush = (HBRUSH) ::CreateSolidBrush(backgroundColor);
|
|
if (backgroundImageURL != "")
|
|
{
|
|
// There's a background that needs to be drawn.
|
|
SetBackgroundImage(LookupImage(backgroundImageURL, NULL));
|
|
}
|
|
|
|
if (GetBackgroundImage() &&
|
|
GetBackgroundImage()->FrameSuccessfullyLoaded())
|
|
{
|
|
PaintBackground(dcPaint.m_hDC, &rcClient, GetBackgroundImage(), 0);
|
|
}
|
|
else
|
|
{
|
|
::FillRect(dcPaint.m_hDC, &rcClient, hRegBrush);
|
|
}
|
|
|
|
VERIFY(::DeleteObject(hRegBrush));
|
|
|
|
for (int i = 0; i < m_nNumButtons; i++)
|
|
{
|
|
m_pButtonArray[i]->GetClientRect(&buttonRect);
|
|
|
|
m_pButtonArray[i]->MapWindowPoints(this, &buttonRect);
|
|
|
|
if(intersectRect.IntersectRect(updateRect, buttonRect))
|
|
{
|
|
MapWindowPoints(m_pButtonArray[i], &intersectRect);
|
|
m_pButtonArray[i]->RedrawWindow(&intersectRect);
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
// CRDFToolbar Messages
|
|
///////////////////////////////////////////////////////////////////////////////////
|
|
|
|
BEGIN_MESSAGE_MAP(CRDFToolbar, CNSToolbar2)
|
|
//{{AFX_MSG_MAP(CNSToolbar2)
|
|
ON_WM_RBUTTONDOWN()
|
|
ON_WM_PAINT()
|
|
//}}AFX_MSG_MAP
|
|
|
|
END_MESSAGE_MAP()
|
|
|
|
void CRDFToolbar::OnRButtonDown(UINT nFlags, CPoint point)
|
|
{
|
|
HT_SetSelectedView(HT_GetPane(GetHTView()), GetHTView());
|
|
HT_SetSelection(HT_TopNode(GetHTView()));
|
|
|
|
m_MenuCommandMap.Clear();
|
|
HT_Cursor theCursor = HT_NewContextualMenuCursor(m_ToolbarView, PR_FALSE, PR_TRUE);
|
|
CMenu menu;
|
|
ClientToScreen(&point);
|
|
if (menu.CreatePopupMenu() != 0 && theCursor != NULL)
|
|
{
|
|
// We have a cursor. Attempt to iterate
|
|
HT_MenuCmd theCommand;
|
|
while (HT_NextContextMenuItem(theCursor, &theCommand))
|
|
{
|
|
char* menuName = HT_GetMenuCmdName(theCommand);
|
|
if (theCommand == HT_CMD_SEPARATOR)
|
|
menu.AppendMenu(MF_SEPARATOR);
|
|
else
|
|
{
|
|
// Add the command to our command map
|
|
CRDFMenuCommand* rdfCommand = new CRDFMenuCommand(menuName, theCommand);
|
|
int index = m_MenuCommandMap.AddCommand(rdfCommand);
|
|
menu.AppendMenu(MF_ENABLED, index+FIRST_HT_MENU_ID, menuName);
|
|
}
|
|
}
|
|
HT_DeleteCursor(theCursor);
|
|
|
|
menu.TrackPopupMenu(TPM_LEFTALIGN | TPM_RIGHTBUTTON, point.x, point.y, this, NULL);
|
|
|
|
menu.DestroyMenu();
|
|
}
|
|
}
|
|
|
|
BOOL CRDFToolbar::OnCommand( WPARAM wParam, LPARAM lParam )
|
|
{
|
|
if (wParam >= FIRST_HT_MENU_ID && wParam <= LAST_HT_MENU_ID)
|
|
{
|
|
// A selection was made from the context menu.
|
|
// Use the menu map to get the HT command value
|
|
CRDFMenuCommand* theCommand = (CRDFMenuCommand*)(m_MenuCommandMap.GetCommand((int)wParam-FIRST_HT_MENU_ID));
|
|
if (theCommand)
|
|
{
|
|
HT_MenuCmd htCommand = theCommand->GetHTCommand();
|
|
HT_DoMenuCmd(HT_GetPane(m_ToolbarView), htCommand);
|
|
}
|
|
return TRUE;
|
|
}
|
|
return((BOOL)GetParentFrame()->SendMessage(WM_COMMAND, wParam, lParam));
|
|
}
|
|
|
|
static char *htButtonStyleProperties[] = {
|
|
"picturesAndText", "pictures", "text"
|
|
};
|
|
|
|
const char * CRDFToolbar::HTDescriptorFromStyle(int style)
|
|
{
|
|
XP_ASSERT(style == nPicAndTextStyle || style == nPicStyle || style == nTextStyle);
|
|
return htButtonStyleProperties[style];
|
|
}
|
|
|
|
int CRDFToolbar::StyleFromHTDescriptor(const char *descriptor)
|
|
{
|
|
int ctr;
|
|
for (ctr = nPicAndTextStyle; ctr <= nTextStyle; ctr++)
|
|
if (strcmp(descriptor, htButtonStyleProperties[ctr]) == 0)
|
|
return ctr;
|
|
XP_ASSERT(0);
|
|
return -1;
|
|
}
|
|
|
|
// ==========================================================
|
|
// CRDFDragToolbar
|
|
// Contains a single RDF toolbar.
|
|
// Handles drawing of backgrounds etc. etc.
|
|
// ==========================================================
|
|
|
|
BEGIN_MESSAGE_MAP(CRDFDragToolbar, CDragToolbar)
|
|
//{{AFX_MSG_MAP(CRDFToolbarButton)
|
|
ON_WM_PAINT()
|
|
//}}AFX_MSG_MAP
|
|
END_MESSAGE_MAP()
|
|
|
|
#define OPEN_BUTTON_WIDTH 9
|
|
#define CLOSED_BUTTON_WIDTH 40
|
|
#define CLOSED_BUTTON_HEIGHT 10
|
|
#define DT_RIGHT_MARGIN 1 //Margin between end of toolbar and right border of Navigator
|
|
#define DRAGGING_BORDER_HEIGHT 2
|
|
#define NOTOOL -1
|
|
|
|
#define HTAB_TOP_START 0
|
|
#define HTAB_TOP_HEIGHT 7
|
|
#define HTAB_MIDDLE_START 8
|
|
#define HTAB_MIDDLE_HEIGHT 6
|
|
#define HTAB_BOTTOM_START 15
|
|
#define HTAB_BOTTOM_HEIGHT 3
|
|
|
|
extern HBITMAP m_hTabBitmap;
|
|
|
|
int CRDFDragToolbar::Create(CWnd *pParent, CToolbarWindow *pToolbar)
|
|
{
|
|
int rtnval = CDragToolbar::Create(pParent, pToolbar);
|
|
|
|
if (rtnval)
|
|
{
|
|
char *data;
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)m_pToolbar->GetToolbar();
|
|
HT_Resource top = HT_TopNode(pToolbar->GetHTView());
|
|
HT_GetNodeData(top, gNavCenter->toolbarCollapsed, HT_COLUMN_STRING, (void **)&data);
|
|
if (data)
|
|
{
|
|
if (data[0] == 'y' || data[0] == 'Y')
|
|
SetOpen(FALSE);
|
|
}
|
|
}
|
|
return rtnval;
|
|
}
|
|
|
|
void CRDFDragToolbar::SetOpen(BOOL bIsOpen)
|
|
{
|
|
CDragToolbar::SetOpen(bIsOpen);
|
|
CopySettingsToRDF();
|
|
}
|
|
|
|
void CRDFDragToolbar::BeActiveToolbar()
|
|
{
|
|
CopySettingsToRDF();
|
|
}
|
|
|
|
void CRDFDragToolbar::CopySettingsToRDF(void)
|
|
{
|
|
char *data = GetOpen() ? "no" : "yes";
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)m_pToolbar->GetToolbar();
|
|
HT_Resource top = HT_TopNode(pToolbar->GetHTView());
|
|
HT_SetNodeData(top, gNavCenter->toolbarCollapsed, HT_COLUMN_STRING, data);
|
|
}
|
|
|
|
void CRDFDragToolbar::OnPaint(void)
|
|
{
|
|
CPaintDC dcPaint(this); // device context for painting
|
|
CRect rect;
|
|
|
|
GetClientRect(&rect);
|
|
|
|
// Use our toolbar background color (or background image TODO)
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)m_pToolbar->GetToolbar();
|
|
|
|
HBRUSH brFace = (HBRUSH) ::CreateSolidBrush(pToolbar->GetBackgroundColor());
|
|
|
|
::FillRect(dcPaint.m_hDC, &rect, brFace);
|
|
|
|
VERIFY(::DeleteObject(brFace));
|
|
|
|
CDC *pDC = &dcPaint;
|
|
HPALETTE hOldPal = ::SelectPalette(pDC->m_hDC, WFE_GetUIPalette(GetParentFrame()), FALSE);
|
|
|
|
if(m_hTabBitmap != NULL)
|
|
{
|
|
|
|
// Create a scratch DC and select our bitmap into it.
|
|
CDC * pBmpDC = new CDC;
|
|
pBmpDC->CreateCompatibleDC(pDC);
|
|
|
|
|
|
HBITMAP hOldBmp = (HBITMAP)::SelectObject(pBmpDC->m_hDC ,m_hTabBitmap);
|
|
HPALETTE hOldPalette = ::SelectPalette(pBmpDC->m_hDC, WFE_GetUIPalette(NULL), TRUE);
|
|
::RealizePalette(pBmpDC->m_hDC);
|
|
CPoint bitmapStart(!m_bMouseInTab ? 0 : OPEN_BUTTON_WIDTH ,HTAB_TOP_START);
|
|
|
|
//First do top of the tab
|
|
FEU_TransBlt( pDC->m_hDC, 0, 0, OPEN_BUTTON_WIDTH, HTAB_TOP_HEIGHT,
|
|
pBmpDC->m_hDC, bitmapStart.x, bitmapStart.y,WFE_GetUIPalette(NULL),
|
|
GetSysColor(COLOR_BTNFACE));
|
|
|
|
//Now do the middle portion of the tab
|
|
int y = HTAB_TOP_HEIGHT;
|
|
|
|
bitmapStart.y = HTAB_MIDDLE_START;
|
|
|
|
while(y < rect.bottom - HTAB_BOTTOM_HEIGHT)
|
|
{
|
|
FEU_TransBlt(pDC->m_hDC, 0, y, OPEN_BUTTON_WIDTH, HTAB_MIDDLE_HEIGHT,
|
|
pBmpDC->m_hDC, bitmapStart.x, bitmapStart.y, WFE_GetUIPalette(NULL),
|
|
GetSysColor(COLOR_BTNFACE));
|
|
|
|
y += HTAB_MIDDLE_HEIGHT;
|
|
|
|
}
|
|
|
|
// Now do the bottom of the tab
|
|
y = rect.bottom - HTAB_BOTTOM_HEIGHT;
|
|
|
|
bitmapStart.y = HTAB_BOTTOM_START;
|
|
|
|
FEU_TransBlt(pDC->m_hDC, 0, y, OPEN_BUTTON_WIDTH, HTAB_BOTTOM_HEIGHT,
|
|
pBmpDC->m_hDC, bitmapStart.x, bitmapStart.y, WFE_GetUIPalette(NULL),
|
|
GetSysColor(COLOR_BTNFACE));
|
|
|
|
|
|
// Cleanup
|
|
::SelectObject(pBmpDC->m_hDC, hOldBmp);
|
|
::SelectPalette(pBmpDC->m_hDC, hOldPalette, TRUE);
|
|
::SelectPalette(pDC->m_hDC, hOldPal, TRUE);
|
|
pBmpDC->DeleteDC();
|
|
delete pBmpDC;
|
|
}
|
|
|
|
if(m_bDragging)
|
|
{
|
|
CBrush brush(RGB(0, 0, 0));
|
|
CBrush *pOldBrush = (CBrush*)dcPaint.SelectObject(&brush);
|
|
|
|
//rect.left += OPEN_BUTTON_WIDTH;
|
|
|
|
dcPaint.FrameRect(&rect, &brush);
|
|
|
|
dcPaint.SelectObject(pOldBrush);
|
|
brush.DeleteObject();
|
|
}
|
|
}
|
|
|
|
// ==========================================================
|
|
// CRDFToolbarHolder
|
|
// The container of all the toolbars
|
|
// ==========================================================
|
|
|
|
CRDFToolbarHolder::CRDFToolbarHolder(int maxToolbars, CFrameWnd* pParentWindow)
|
|
:CCustToolbar(maxToolbars)
|
|
{
|
|
m_pCachedParentWindow = pParentWindow;
|
|
m_pCurrentButton = NULL;
|
|
}
|
|
|
|
CRDFToolbarHolder::~CRDFToolbarHolder()
|
|
{
|
|
if (m_ToolbarPane)
|
|
{
|
|
HT_Pane oldPane = m_ToolbarPane;
|
|
m_ToolbarPane = NULL;
|
|
HT_DeletePane(oldPane);
|
|
}
|
|
}
|
|
|
|
void CRDFToolbarHolder::DrawSeparator(int i, HDC hDC, int nStartX, int nEndX, int nStartY,
|
|
BOOL bToolbarSeparator)
|
|
{
|
|
// Get the toolbar and force it to compute its colors.
|
|
COLORREF highlightColor = ::GetSysColor(COLOR_BTNHIGHLIGHT);
|
|
COLORREF shadowColor = ::GetSysColor(COLOR_BTNSHADOW);
|
|
|
|
if (i < m_nNumOpen)
|
|
{
|
|
if (m_pToolbarArray[i] != NULL)
|
|
{
|
|
CRDFToolbar* pToolbar = (CRDFToolbar*)(m_pToolbarArray[i]->GetToolbar());
|
|
pToolbar->ComputeColorsForSeparators();
|
|
|
|
shadowColor = pToolbar->GetShadowColor();
|
|
highlightColor = pToolbar->GetHighlightColor();
|
|
}
|
|
}
|
|
|
|
HPEN pen = ::CreatePen(PS_SOLID, 1, shadowColor);
|
|
HPEN pOldPen = (HPEN)::SelectObject(hDC, pen);
|
|
|
|
::MoveToEx(hDC, nStartX, nStartY, NULL);
|
|
::LineTo(hDC, nEndX, nStartY);
|
|
|
|
::SelectObject(hDC, pOldPen);
|
|
|
|
::DeleteObject(pen);
|
|
|
|
if(bToolbarSeparator)
|
|
{
|
|
pen = ::CreatePen(PS_SOLID, 1, highlightColor);
|
|
}
|
|
else
|
|
{
|
|
pen = ::CreatePen(PS_SOLID, 1, highlightColor);
|
|
}
|
|
|
|
::SelectObject(hDC, pen);
|
|
|
|
::MoveToEx(hDC, nStartX, nStartY + 1, NULL);
|
|
|
|
::LineTo(hDC, nEndX, nStartY + 1);
|
|
|
|
::SelectObject(hDC, pOldPen);
|
|
::DeleteObject(pen);
|
|
}
|
|
|
|
void CRDFToolbarHolder::InitializeRDFData()
|
|
{
|
|
HT_Notification ns = new HT_NotificationStruct;
|
|
XP_BZERO(ns, sizeof(HT_NotificationStruct));
|
|
ns->notifyProc = toolbarNotifyProcedure;
|
|
ns->data = this;
|
|
|
|
// Construct the pane and give it our notification struct
|
|
HT_Pane newPane = HT_NewToolbarPane(ns);
|
|
if (newPane)
|
|
{
|
|
SetHTPane(newPane);
|
|
HT_SetPaneFEData(newPane, this);
|
|
}
|
|
}
|
|
|
|
CIsomorphicCommandMap* CIsomorphicCommandMap::InitializeCommandMap(const CString& initType)
|
|
{
|
|
CIsomorphicCommandMap* result = new CIsomorphicCommandMap();
|
|
|
|
if (initType == "Browser Commands")
|
|
{
|
|
// Enter the builtin browser commands into the map.
|
|
result->AddItem("command:back", ID_NAVIGATE_BACK);
|
|
result->AddItem("command:forward", ID_NAVIGATE_FORWARD);
|
|
result->AddItem("command:reload", ID_NAVIGATE_RELOAD);
|
|
result->AddItem("command:home", ID_GO_HOME);
|
|
result->AddItem("command:print", ID_FILE_PRINT);
|
|
result->AddItem("command:stop", ID_NAVIGATE_INTERRUPT);
|
|
}
|
|
else if (initType == "Command Toolbar Bitmap Indices")
|
|
{
|
|
result->AddItem("command:back", 0);
|
|
result->AddItem("command:forward", 1);
|
|
result->AddItem("command:reload", 2);
|
|
result->AddItem("command:home", 3);
|
|
result->AddItem("command:print", 7);
|
|
result->AddItem("command:stop", 11);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void CIsomorphicCommandMap::AddItem(CString xpItem, UINT feResource)
|
|
{
|
|
char buffer[20];
|
|
_itoa((int)feResource, buffer, 10);
|
|
mapFromXPToFE.SetAt(xpItem, CString(buffer));
|
|
mapFromFEToXP.SetAt(CString(buffer), xpItem);
|
|
}
|
|
|
|
void CIsomorphicCommandMap::RemoveXPItem(CString xpItem)
|
|
{
|
|
CString result;
|
|
mapFromXPToFE.Lookup(xpItem, result);
|
|
mapFromXPToFE.RemoveKey(xpItem);
|
|
mapFromFEToXP.RemoveKey(result);
|
|
}
|
|
|
|
void CIsomorphicCommandMap::RemoveFEItem(UINT feResource)
|
|
{
|
|
char buffer[20];
|
|
_itoa((int)feResource, buffer, 10);
|
|
CString resource(buffer);
|
|
CString result;
|
|
mapFromFEToXP.Lookup(resource, result);
|
|
mapFromXPToFE.RemoveKey(result);
|
|
mapFromFEToXP.RemoveKey(resource);
|
|
}
|
|
|
|
UINT CIsomorphicCommandMap::GetFEResource(CString xpItem)
|
|
{
|
|
CString result;
|
|
mapFromXPToFE.Lookup(xpItem, result);
|
|
return (atoi(result));
|
|
}
|
|
|
|
CString CIsomorphicCommandMap::GetXPResource(UINT feResource)
|
|
{
|
|
char buffer[20];
|
|
_itoa((int)feResource, buffer, 10);
|
|
CString resource(buffer);
|
|
CString result;
|
|
mapFromFEToXP.Lookup(resource, result);
|
|
return result;
|
|
}
|