#include "main.h" #include "./toolbar.h" #include "./graphics.h" #include "./resource.h" #include "./toolbarItem.h" #include "./toolbarStatic.h" #include "./toolbarButton.h" #include "./toolbarRating.h" #include "./toolbarProgress.h" #include "./toolbarAddress.h" #include "../winamp/wa_dlg.h" #include "../Plugins/General/gen_ml/ml_ipc_0313.h" #include "./ifc_imageloader.h" #include "./ifc_skinhelper.h" #include "./ifc_omservice.h" #include "./ifc_omservicecommand.h" #include "./menu.h" #include "./ifc_wasabihelper.h" #include "./browserUiCommon.h" #include #include #include #include #include #define TOOLBAR_SPACE_LEFT 0 #define TOOLBAR_SPACE_TOP 4 #define TOOLBAR_SPACE_RIGHT 1 #define TOOLBAR_SPACE_BOTTOM 3 #define TOOLBAR_HIDDENHEIGHT 3 #define TOOLBAR_ICONSTATE_NORMAL 0 #define TOOLBAR_ICONSTATE_HIGHLIGHTED 1 #define TOOLBAR_ICONSTATE_PRESSED 2 #define TOOLBAR_ICONSTATE_DISABLED 3 #define TOOLBAR_ICONSTATE_COUNT 4 #define TBS_MENULOOP 0x00000001 #define TBS_HIDDEN 0x00000002 #define TBS_HIDETIMERSET 0x00000004 #define TBS_NOFOCUSRECT 0x00000008 #define ICON_NONE (-1) #define ICON_CHEVRON (-2) // - resolved on the fly #define ICON_SEPARATOR 0 #define ICON_CHEVRON_BOTTOM 1 #define ICON_CHEVRON_TOP 2 #define ICON_BACK 3 #define ICON_FORWARD 4 #define ICON_REFRESH 5 #define ICON_STOP 6 #define ICON_HOME 7 #define ICON_ERROR 8 #define ICON_LOCK 9 #define ICON_HISTORY 10 #define ICON_ADDRESSBAR 11 #define ICON_SEPARATOR_WIDTH 6 #define ICON_CHEVRON_WIDTH 13 #define ICON_HISTORY_WIDTH 9 #define ICON_ARROW_WIDTH 17 #define TOOLBRUSH_BACK 0 #define TOOLBRUSH_FRAME 1 #define TOOLBRUSH_ITEMBK 2 #define TOOLBRUSH_LAST TOOLBRUSH_ITEMBK #define ID_CHEVRON_CLICKED 1 #define TOOLBAR_BKCOLOR WADLG_LISTHEADER_BGCOLOR #define TOOLBAR_FGCOLOR WADLG_LISTHEADER_FONTCOLOR #define EDIT_ALPHA 50 #define TEXT_ALPHA 127 #define HIGHLIGHT_ALPHA 160 #define TIMER_AUTOHIDE_ID 14 #define TIMER_AUTOHIDE_DELAY 300 #define AUTOHIDE_ANIMATETIME 200 typedef std::vector ItemList; typedef struct __TOOLBAR { UINT ref; UINT flags; HBRUSH szBrushes[TOOLBRUSH_LAST + 1]; HIMAGELIST imageList; HFONT textFont; ItemList *items; ToolbarItem *chevron; ToolbarItem *pressed; ToolbarItem *highlighted; UINT resolveCache; HFONT ownedFont; HWND hTooltip; COLORREF rgbBk; COLORREF rgbFg; COLORREF rgbText; COLORREF rgbHilite; COLORREF rgbFrame; COLORREF rgbEdit; COLORREF rgbEditBk; size_t iFocused; HWND hBrowser; TOOLBARTEXTMETRIC textMetric; } TOOLBAR; typedef struct __TOOLBARMOUSEHOOK { HHOOK hHook; HWND hwnd; } TOOLBARMOUSEHOOK; static size_t tlsIndex = TLS_OUT_OF_INDEXES; typedef ToolbarItem *(CALLBACK *TOOLBARITEMFACTORY)(ToolbarItem::Template * /*itemTemplate*/); typedef struct __TOOLBARITEMCLASS { LPCSTR name; TOOLBARITEMFACTORY creator; LPCWSTR text; LPCWSTR description; INT iconId; INT commandId; UINT style; } TOOLBARITEMCLASS; #define REGISTER_ITEM(__name, __creator, __textId, __descriptionId, __iconId, __commandId, __style)\ { (__name), (__creator), MAKEINTRESOURCE(__textId),\ MAKEINTRESOURCE(__descriptionId), (__iconId), (__commandId), (__style) } #define REGISTER_STATIC(__name, __textId, __descriptionId, __iconId, __style)\ REGISTER_ITEM(__name, ToolbarStatic::CreateInstance, __textId, __descriptionId, __iconId, 0, __style) #define REGISTER_BUTTON(__name, __textId, __descriptionId, __iconId, __commandId, __style)\ REGISTER_ITEM(__name, ToolbarButton::CreateInstance, __textId, __descriptionId, __iconId, __commandId, __style) #define REGISTER_RATING(__name, __textId, __descriptionId, __style)\ REGISTER_ITEM(__name, ToolbarRating::CreateInstance, __textId, __descriptionId, ICON_NONE, 0, __style) #define REGISTER_PROGRESS(__name, __textId, __descriptionId, __style)\ REGISTER_ITEM(__name, ToolbarProgress::CreateInstance, __textId, __descriptionId, ICON_NONE, 0, __style) #define REGISTER_ADDRESSBAR(__name, __style)\ REGISTER_ITEM(__name, ToolbarAddress::CreateInstance, NULL, NULL, ICON_NONE, 0, __style) const static TOOLBARITEMCLASS szRegisteredToolItems[] = { REGISTER_STATIC(TOOLITEM_SEPARATOR, IDS_SEPARATOR, 0, ICON_SEPARATOR, ToolbarStatic::styleSeparator), REGISTER_STATIC(TOOLITEM_SPACE, IDS_SPACE, 0, ICON_NONE, ToolbarStatic::styleSpacer), REGISTER_STATIC(TOOLITEM_FLEXSPACE, IDS_FLEXSPACE, 0, ICON_NONE, ToolbarStatic::styleSpacer | ToolbarItem::styleFlexible), REGISTER_RATING(TOOLITEM_USERRATING, IDS_RATED, NULL, ToolbarItem::styleWantKey), REGISTER_PROGRESS(TOOLITEM_DOWNLOADPROGRESS, NULL, NULL, ToolbarItem::stateDisabled), REGISTER_ADDRESSBAR(TOOLITEM_ADDRESSBAR, ToolbarItem::styleFlexible | ToolbarItem::styleWantKey | ToolbarItem::styleTabstop), REGISTER_BUTTON(TOOLITEM_CHEVRON, IDS_MORE, IDS_MORE_DESCRIPTION, ICON_CHEVRON, ID_CHEVRON_CLICKED, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_HOME, IDS_HOME, IDS_HOME_DESCRIPTION, ICON_HOME, ID_NAVIGATION_HOME, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_BACK, IDS_BACK, IDS_BACK_DESCRIPTION, ICON_BACK, ID_NAVIGATION_BACK, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_FORWARD, IDS_FORWARD, IDS_FORWARD_DESCRIPTION, ICON_FORWARD, ID_NAVIGATION_FORWARD, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_STOP, IDS_STOP, IDS_STOP_DESCRIPTION, ICON_STOP, ID_NAVIGATION_STOP, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_REFRESH, IDS_REFRESH, IDS_REFRESH_DESCRIPTION, ICON_REFRESH, ID_NAVIGATION_REFRESH, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_HISTORY, IDS_HISTORY, IDS_HISTORY_DESCRIPTION, ICON_HISTORY, ID_NAVIGATION_HISTORY, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_CMDLINK_INFO, IDS_SERVICE_GETINFO, IDS_SERVICE_GETINFO_DESCRIPTION, ICON_NONE, ID_SERVICE_GETINFO, ToolbarItem::styleWantKey | ToolbarButton::styleCommandLink), REGISTER_BUTTON(TOOLITEM_CMDLINK_REPORT, IDS_SERVICE_REPORT, IDS_SERVICE_REPORT_DESCRIPTION, ICON_NONE, ID_SERVICE_REPORT, ToolbarItem::styleWantKey | ToolbarButton::styleCommandLink), REGISTER_BUTTON(TOOLITEM_CMDLINK_UNSUBSCRIBE, IDS_SERVICE_UNSUBSCRIBE, IDS_SERVICE_UNSUBSCRIBE_DESCRIPTION, ICON_NONE, ID_SERVICE_UNSUBSCRIBE, ToolbarItem::styleWantKey | ToolbarButton::styleCommandLink), REGISTER_BUTTON(TOOLITEM_BUTTON_SECURECONNECTION, IDS_SECURE_CONNECTION, IDS_SECURE_CONNECTION, ICON_LOCK, ID_BROWSER_SECURECONNECTION, ToolbarItem::styleWantKey), REGISTER_BUTTON(TOOLITEM_BUTTON_SCRIPTERROR, IDS_SCRIPT_ERROR, IDS_SCRIPT_ERROR_DESCRIPTION, ICON_ERROR, ID_BROWSER_SCRIPTERROR, ToolbarItem::styleWantKey), }; #define GetToolbar(__hwnd) ((TOOLBAR*)(LONG_PTR)(LONGX86)GetWindowLongPtr((__hwnd), 0)) static LRESULT CALLBACK Toolbar_WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam); static LRESULT CALLBACK Toolbar_MouseHook(INT code, WPARAM wParam, LPARAM lParam); typedef struct __TOOLBARITEMINSERTREC { LPCSTR name; UINT style; } TOOLBARITEMINSERTREC; BOOL Toolbar_RegisterClass(HINSTANCE hInstance) { WNDCLASS wc; ATOM klassAtom; ifc_wasabihelper *wasabi; if (GetClassInfo(hInstance, NWC_ONLINEMEDIATOOLBAR, &wc)) return TRUE; ZeroMemory(&wc, sizeof(WNDCLASS)); wc.hInstance = hInstance; wc.lpszClassName = NWC_ONLINEMEDIATOOLBAR; wc.lpfnWndProc = Toolbar_WindowProc; wc.style = 0; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = NULL; wc.cbWndExtra = sizeof(TOOLBAR*); klassAtom = RegisterClassW(&wc); if (0 == klassAtom) return FALSE; if (SUCCEEDED(Plugin_GetWasabiHelper(&wasabi))) { api_application *application; if (SUCCEEDED(wasabi->GetApplicationApi(&application))) { application->DirectMouseWheel_RegisterSkipClass(klassAtom); application->Release(); } wasabi->Release(); } return TRUE; } static BOOL Toolbar_ClearItems(TOOLBAR *toolbar) { if (NULL == toolbar) return FALSE; if (NULL == toolbar->items) return TRUE; size_t index = toolbar->items->size(); while(index--) { ToolbarItem *item = toolbar->items->at(index); if (NULL != item) item->Release(); } toolbar->items->clear(); return TRUE; } static ULONG Toolbar_AddRef(TOOLBAR *toolbar) { if (NULL == toolbar) return 0; return InterlockedIncrement((LONG*)&toolbar->ref); } static ULONG Toolbar_Release(TOOLBAR *toolbar) { if (0 == toolbar || 0 == toolbar->ref) return 0; LONG r = InterlockedDecrement((LONG*)&toolbar->ref); if (0 == r) { if (NULL != toolbar->hTooltip) DestroyWindow(toolbar->hTooltip); if (NULL != toolbar->chevron) toolbar->chevron->Release(); if (NULL != toolbar->items) { Toolbar_ClearItems(toolbar); delete(toolbar->items); } if (NULL != toolbar->imageList) ImageList_Destroy(toolbar->imageList); if (NULL != toolbar->ownedFont) DeleteObject(toolbar->ownedFont); free(toolbar); } return r; } static const TOOLBARITEMCLASS *Toolbar_FindClass(LPCSTR pszName) { if (NULL == pszName) return NULL; for (INT i =0; i < ARRAYSIZE(szRegisteredToolItems); i++) { if (CSTR_EQUAL == CompareStringA(CSTR_INVARIANT, NORM_IGNORECASE, szRegisteredToolItems[i].name, -1, pszName, -1)) { return &szRegisteredToolItems[i]; } } return NULL; } static ToolbarItem* Toolbar_CreateItem(LPCSTR pszName, UINT styleOverride) { const TOOLBARITEMCLASS *iClass = Toolbar_FindClass(pszName); if (NULL == iClass || NULL == iClass->creator) return NULL; ToolbarItem::Template iTemplate; ZeroMemory(&iTemplate, sizeof(ToolbarItem::Template)); iTemplate.name = iClass->name; iTemplate.text = iClass->text; iTemplate.description= iClass->description; iTemplate.iconId = iClass->iconId; iTemplate.commandId = iClass->commandId; iTemplate.style = iClass->style | styleOverride; return iClass->creator(&iTemplate); } static BOOL Toolbar_GetClientRect(HWND hwnd, RECT *prc) { if (!GetClientRect(hwnd, prc)) return FALSE; prc->left += TOOLBAR_SPACE_LEFT; prc->right -= TOOLBAR_SPACE_RIGHT; if (0 != (TBS_BOTTOMDOCK & GetWindowLongPtr(hwnd, GWL_STYLE))) { prc->top += TOOLBAR_SPACE_TOP; prc->bottom -= TOOLBAR_SPACE_BOTTOM; } else { prc->top += TOOLBAR_SPACE_BOTTOM; prc->bottom -= TOOLBAR_SPACE_TOP; } return TRUE; } static void Toolbar_UpdateColorTable(HWND hwnd, ifc_skinhelper *skinHelper) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; if (NULL == skinHelper || FAILED(skinHelper->GetColor(TOOLBAR_BKCOLOR, &toolbar->rgbBk))) toolbar->rgbBk = GetSysColor(COLOR_WINDOW); if (NULL == skinHelper || FAILED(skinHelper->GetColor(TOOLBAR_FGCOLOR, &toolbar->rgbFg))) toolbar->rgbFg = GetSysColor(COLOR_WINDOWTEXT); COLORREF rgbItemBk; if (NULL == skinHelper || FAILED(skinHelper->GetColor(WADLG_ITEMBG, &rgbItemBk))) rgbItemBk = GetSysColor(COLOR_WINDOW); COLORREF rgbItem; if (NULL == skinHelper || FAILED(skinHelper->GetColor(WADLG_ITEMFG, &rgbItem))) rgbItem = GetSysColor(COLOR_WINDOWTEXT); if (NULL != skinHelper) { const INT szFrameColors[] = { WADLG_HILITE, WADLG_LISTHEADER_FRAME_MIDDLECOLOR, WADLG_LISTHEADER_FRAME_BOTTOMCOLOR, WADLG_LISTHEADER_FRAME_TOPCOLOR, }; for (INT i = 0; i < ARRAYSIZE(szFrameColors); i++) { if (SUCCEEDED(skinHelper->GetColor(szFrameColors[i], &toolbar->rgbFrame))) toolbar->rgbFrame = GetSysColor(COLOR_GRAYTEXT); INT distance = GetColorDistance(toolbar->rgbFrame, rgbItemBk); if (distance < 0) distance = -distance; if (distance >= 40) break; } } else toolbar->rgbFrame = GetSysColor(COLOR_GRAYTEXT); toolbar->rgbEdit = BlendColors(toolbar->rgbBk, rgbItem, EDIT_ALPHA); toolbar->rgbEditBk = BlendColors(toolbar->rgbBk, rgbItemBk, EDIT_ALPHA); toolbar->rgbText = BlendColors(toolbar->rgbFg, toolbar->rgbBk, TEXT_ALPHA); toolbar->rgbHilite = BlendColors(toolbar->rgbFg, toolbar->rgbBk, HIGHLIGHT_ALPHA); } static void Toolbar_UpdateTextMetrics(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; ZeroMemory(&toolbar->textMetric, sizeof(TOOLBARTEXTMETRIC)); RECT clientRect; if (!Toolbar_GetClientRect(hwnd, &clientRect)) return; INT iconCX, iconCY; if (NULL == toolbar->imageList || FALSE == ImageList_GetIconSize(toolbar->imageList, &iconCX, &iconCY)) { iconCX = 0; iconCY = 0; } else { iconCY = iconCY / TOOLBAR_ICONSTATE_COUNT; } HDC hdc = GetDCEx(hwnd, NULL, DCX_CACHE | DCX_NORESETATTRS); if (NULL == hdc) return; HFONT originalFont = (HFONT)SelectObject(hdc, toolbar->textFont); TEXTMETRIC tm; if (GetTextMetrics(hdc, &tm)) { toolbar->textMetric.height = tm.tmHeight; toolbar->textMetric.aveCharWidth = tm.tmAveCharWidth; toolbar->textMetric.overhang = tm.tmOverhang; INT clientHeight = clientRect.bottom - clientRect.top; if (tm.tmHeight >= iconCY) { toolbar->textMetric.baseY = tm.tmAscent - tm.tmInternalLeading; INT t = clientHeight - toolbar->textMetric.baseY; toolbar->textMetric.origY = clientRect.top + t/2; if (0 != t%2) { toolbar->textMetric.origY += 1; toolbar->textMetric.baseY += 1; } } else { toolbar->textMetric.baseY = (clientHeight - iconCY)/2; if (tm.tmDescent > toolbar->textMetric.baseY) toolbar->textMetric.baseY = tm.tmDescent; //toolbar->textMetric.baseY += 8; toolbar->textMetric.baseY = clientHeight - toolbar->textMetric.baseY; toolbar->textMetric.origY = (clientRect.top + toolbar->textMetric.baseY + tm.tmDescent) - tm.tmHeight; toolbar->textMetric.baseY -= toolbar->textMetric.origY; } } SelectObject(hdc, originalFont); ReleaseDC(hwnd, hdc); } static HIMAGELIST Toolbar_LoadImagelist(TOOLBAR *toolbar, HINSTANCE hInstance, LPCWSTR pszPath, INT iconWidth, INT iconStatesCount) { if (iconWidth <= 0 || iconStatesCount <= 0 || NULL == pszPath) return NULL; HBITMAP bitmap = NULL; ifc_omimageloader *loader; if (SUCCEEDED(Plugin_QueryImageLoader(hInstance, pszPath, FALSE, &loader))) { loader->LoadBitmap(&bitmap, NULL, NULL); loader->Release(); } DIBSECTION dib; if (NULL == bitmap || sizeof(DIBSECTION) != GetObject(bitmap, sizeof(DIBSECTION), &dib)) { if (NULL != bitmap) DeleteObject(bitmap); return NULL; } if (dib.dsBm.bmHeight < 0) dib.dsBm.bmHeight = -dib.dsBm.bmHeight; Image_Colorize((BYTE*)dib.dsBm.bmBits, dib.dsBm.bmWidth, dib.dsBm.bmHeight, dib.dsBm.bmBitsPixel, ColorAdjustLuma(toolbar->rgbBk, -140, TRUE), toolbar->rgbFg, FALSE); Image_BlendOnColorEx((BYTE*)dib.dsBm.bmBits, dib.dsBm.bmWidth, dib.dsBm.bmHeight, 0, 0, dib.dsBm.bmWidth, dib.dsBm.bmHeight, dib.dsBm.bmBitsPixel, FALSE, toolbar->rgbBk); if (iconWidth > dib.dsBm.bmWidth) iconWidth = dib.dsBm.bmWidth; INT iconHeight = dib.dsBm.bmHeight / iconStatesCount; INT initial = dib.dsBm.bmWidth / iconWidth; HIMAGELIST himl = ImageList_Create(iconWidth, iconHeight * iconStatesCount, ILC_COLOR24, initial, 1); if (NULL != himl) { INT index = ImageList_Add(himl, bitmap, NULL); if (-1 == index) { ImageList_Destroy(himl); himl = NULL; } } DeleteObject(bitmap); return himl; } static LONG Toolbar_ShowChevron(HWND hwnd, const RECT *prcClient) // return client cx change { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->chevron && toolbar->chevron->IsRectEmpty()) { RECT chevronRect; CopyRect(&chevronRect, prcClient); if (toolbar->chevron->AdjustRect(hwnd, &chevronRect)) { chevronRect.left = prcClient->right - (chevronRect.right - chevronRect.left); chevronRect.right = prcClient->right; if (toolbar->chevron->SetRect(&chevronRect)) return (chevronRect.left - prcClient->right); } } return 0; } static ToolbarItem *Toolbar_GetItem(TOOLBAR *toolbar, size_t index) { if (NULL == toolbar || NULL == toolbar->items) return NULL; size_t count = toolbar->items->size(); if (index < count) return toolbar->items->at(index); if (index == count) return toolbar->chevron; return NULL; } static BOOL Toolbar_IsItemAcceptFocus(ToolbarItem *item, BOOL fTabstopOnly) { if (NULL == item) return FALSE; UINT itemStyle = item->GetStyle(); if (0 != ((ToolbarItem::stateDisabled | ToolbarItem::stateHidden) & itemStyle)) return FALSE; UINT mask = ToolbarItem::styleTabstop; if (FALSE == fTabstopOnly) mask |= ToolbarItem::styleWantKey; if (0 == (mask & itemStyle)) return FALSE; return (FALSE == item->IsRectEmpty()); } static void Toolbar_UpdateTabstop(HWND hwnd) { BOOL fEnable = FALSE; UINT windowStyle = GetWindowStyle(hwnd); if (0 != (TBS_TABSTOP & windowStyle)) { fEnable = TRUE; } else { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->items) { size_t index = toolbar->items->size(); while(index--) { if (FALSE != Toolbar_IsItemAcceptFocus(toolbar->items->at(index), TRUE)) { fEnable = TRUE; break; } } } } if (FALSE != fEnable) { if (0 == (WS_TABSTOP & windowStyle)) SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle | WS_TABSTOP); } else { if (0 != (WS_TABSTOP & windowStyle)) { SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle & ~WS_TABSTOP); } } } static INT Toolbar_InsertItemInternal(TOOLBAR *toolbar, ToolbarItem *item, INT insertBefore, HWND hwnd) { if (NULL == toolbar || NULL == toolbar->items || NULL == item) return ITEM_ERR; INT index = ITEM_ERR; if (TBIP_FIRST == insertBefore) { toolbar->items->insert(toolbar->items->begin(), item); index = 0; } else if (TBIP_LAST == insertBefore) { toolbar->items->push_back(item); index = ((INT)toolbar->items->size() - 1); } else { if (insertBefore < 0) insertBefore = 0; else if (insertBefore >= (INT)toolbar->items->size()) insertBefore = (INT)toolbar->items->size(); toolbar->items->insert(toolbar->items->begin() + insertBefore, item); index = insertBefore; } if (ITEM_ERR != index) { item->AddRef(); item->UpdateSkin(hwnd); } return index; } static size_t Toolbar_GetFocusIndex(HWND hwnd, size_t focusIndex, INT searchDirection, BOOL fTabstopOnly) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items || 0 == toolbar->items->size()) return ((size_t)-1); ToolbarItem *item = Toolbar_GetItem(toolbar, focusIndex); if (NULL != item && Toolbar_IsItemAcceptFocus(item, fTabstopOnly)) return focusIndex; if (0 == searchDirection) return ((size_t)-1); if (searchDirection > 0) { size_t count = toolbar->items->size(); size_t i = focusIndex + 1; for (; i < count; i++) { if (Toolbar_IsItemAcceptFocus(toolbar->items->at(i), fTabstopOnly)) return i; } if (i == count && Toolbar_IsItemAcceptFocus(toolbar->chevron, fTabstopOnly)) return i; } else { size_t count = toolbar->items->size(); if (focusIndex == count && Toolbar_IsItemAcceptFocus(toolbar->chevron, fTabstopOnly)) return count; size_t i = focusIndex; if (i <= count) { while(i--) { if (Toolbar_IsItemAcceptFocus(toolbar->items->at(i), fTabstopOnly)) return i; } } } return ((size_t)-1); } static void Toolbar_OffsetItems(ToolbarItem **itemList, INT count, INT offsetX, INT offsetY) { while(count-- > 0) { if (NULL != itemList[count]) itemList[count]->OffsetRect(offsetX, offsetY); } } static void Toolbar_HideItems(ToolbarItem **itemList, INT count, INT *chevronItemCount) { INT chevronItems = 0; while(count-- > 0) { if (NULL != itemList[count]) { if (0 == (ToolbarItem::styleNoChevron & itemList[count]->GetStyle())) chevronItems++; itemList[count]->SetRectEmpty(); } } if (NULL != chevronItemCount) *chevronItemCount = chevronItems; } static LONG Toolbar_CutoffItems(ToolbarItem **itemList, INT count, LONG cutoffCX) { RECT elementRect; LONG left = cutoffCX; ToolbarItem *item; while(count-- > 0) { item = itemList[count]; if (NULL == item) continue; DWORD style = item->GetStyle(); if (0 == (ToolbarItem::styleChevronOnly & style) && (0 == (ToolbarItem::stateHidden & style) || 0 == (ToolbarItem::stylePopup & style)) && item->GetRect(&elementRect)) { if (elementRect.right <= cutoffCX) return elementRect.right; if (0 == count) left = elementRect.left; item->SetRectEmpty(); } } return left; } static LONG Toolbar_CalculateClient(HWND hwnd, ToolbarItem **itemList, INT count, const RECT *prcClient, INT *deltaCX) { ToolbarItem *item; RECT elementRect; INT index; UINT style; LONG elementLeft = prcClient->left; for (index = 0; index < count; index++) { item = itemList[index]; if (NULL != item) { style = item->GetStyle(); if (0 != (ToolbarItem::styleChevronOnly & style) || (0 != (ToolbarItem::stateHidden & style) && 0!= (ToolbarItem::stylePopup & style))) { item->SetRectEmpty(); continue; } CopyRect(&elementRect, prcClient); elementRect.left = elementLeft; if (0 != (ToolbarItem::styleFlexible & style)) { RECT testRect; INT flexMinimum = 0; SetRect(&testRect, elementRect.left, elementRect.top, elementRect.left, elementRect.bottom); if (item->AdjustRect(hwnd, &testRect) && testRect.right > elementRect.left) { flexMinimum = (testRect.right - elementRect.left); } if (index != (count -1)) { INT delta = 0; INT section; if ((elementRect.left + flexMinimum) < prcClient->right) { CopyRect(&testRect, &elementRect); testRect.left += flexMinimum; section = Toolbar_CalculateClient(hwnd, itemList + (index + 1), count - index - 1, &testRect, &delta); } else { section = prcClient->right - (elementRect.left + flexMinimum); } if (section < 0) { // we need to move back item->SetRectEmpty(); elementLeft += (flexMinimum - section); if (index > 0) elementLeft = Toolbar_CutoffItems(itemList, index, elementLeft); return (elementLeft - prcClient->left); } if (NULL != deltaCX) *deltaCX += delta; elementRect.right = elementRect.right - section + delta; if (elementRect.right < elementRect.left) elementRect.right = elementRect.left; } if (item->AdjustRect(hwnd, &elementRect)) { item->SetRect(&elementRect); Toolbar_OffsetItems(itemList + (index + 1), count - index - 1, elementRect.right - elementRect.left - flexMinimum, 0); elementLeft = prcClient->right; break; } } if (item->AdjustRect(hwnd, &elementRect)) { if (elementRect.right > prcClient->right) { INT chevronItems; Toolbar_HideItems(itemList + index, count - index, &chevronItems); if (chevronItems > 0) { LONG delta = Toolbar_ShowChevron(hwnd, prcClient); if (NULL != deltaCX) *deltaCX += delta; elementLeft = (index > 0) ? Toolbar_CutoffItems(itemList, index, prcClient->right + delta) : (prcClient->left + delta); } break; } item->SetRect(&elementRect); elementLeft += (elementRect.right - elementRect.left); } } } return (elementLeft - prcClient->left); } static void Toolbar_UpdateLayout(HWND hwnd, BOOL fRedraw) { UINT windowStyle = GetWindowStyle(hwnd); if (0 != (TBS_LOCKUPDATE & windowStyle)) return; TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; RECT clientRect; if (!Toolbar_GetClientRect(hwnd, &clientRect)) return; Toolbar_UpdateTextMetrics(hwnd); size_t count = (NULL != toolbar->items) ? toolbar->items->size() : 0; if (NULL != toolbar->chevron) { INT chevronOnly = 0; ToolbarItem *item; for (size_t i = 0; i < count; i++) { item = toolbar->items->at(i); UINT style = item->GetStyle(); if (NULL != item && 0 != (ToolbarItem::styleChevronOnly & style) && 0 == (ToolbarItem::stateHidden & style)) { chevronOnly++; } } toolbar->chevron->SetRectEmpty(); if (0 != chevronOnly) clientRect.right += Toolbar_ShowChevron(hwnd, &clientRect); } Toolbar_CalculateClient(hwnd, toolbar->items->size() ? &toolbar->items->at(0) : nullptr, (INT)count, &clientRect, NULL); if (((size_t)-1) != toolbar->iFocused) { size_t focusIndex = Toolbar_GetFocusIndex(hwnd, toolbar->iFocused, -1, FALSE); if (focusIndex != toolbar->iFocused) { ToolbarItem *itemNew, *itemOld; itemOld = Toolbar_GetItem(toolbar, toolbar->iFocused); itemNew = Toolbar_GetItem(toolbar, focusIndex); toolbar->iFocused = focusIndex; if (NULL != itemOld) itemOld->SetFocus(hwnd, itemNew, FALSE); if (NULL != itemNew) itemNew->SetFocus(hwnd, itemOld, TRUE); } } if (FALSE != fRedraw) { RedrawWindow(hwnd, NULL, NULL, RDW_INVALIDATE | RDW_ALLCHILDREN | RDW_NOERASE); } } static ToolbarItem *Toolbar_HitTest(HWND hwnd, POINT pt) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar) { ToolbarItem *item; size_t count = (NULL != toolbar->items) ? toolbar->items->size() : 0; for (size_t i = 0; i <= count; i++) { item = (i < count) ? toolbar->items->at(i) : toolbar->chevron; if (NULL != item && item->PtInItem(pt)) return item; } } return NULL; } static HFONT Toolbar_CreateFont(HFONT skinFont) { LOGFONT lf; if (NULL == skinFont) return NULL; INT skinHeight = (sizeof(LOGFONT) == GetObject(skinFont, sizeof(LOGFONT), &lf)) ? lf.lfHeight : -11; ZeroMemory(&lf, sizeof(LOGFONT)); lf.lfHeight = skinHeight; lf.lfWeight = FW_DONTCARE; lf.lfItalic = FALSE; lf.lfUnderline = FALSE; lf.lfStrikeOut = FALSE; lf.lfCharSet = DEFAULT_CHARSET; lf.lfOutPrecision = OUT_DEFAULT_PRECIS; lf.lfClipPrecision = CLIP_DEFAULT_PRECIS; lf.lfQuality = DEFAULT_QUALITY; lf.lfPitchAndFamily = DEFAULT_PITCH | FF_DONTCARE; StringCchCopy(lf.lfFaceName, ARRAYSIZE(lf.lfFaceName), L"MS Shell Dlg 2"); return CreateFontIndirect(&lf); } static HWND Toolbar_CreateTooltip(HINSTANCE hInstance, HWND hOwner) { HWND hTooltip = CreateWindowExW(WS_EX_TRANSPARENT , TOOLTIPS_CLASS, NULL, WS_POPUP | TTS_ALWAYSTIP, 0, 0, 0, 0, hOwner, NULL, hInstance, NULL); if (NULL == hTooltip) return NULL; SetWindowPos(hTooltip, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE | SWP_NOOWNERZORDER); SendMessage(hTooltip, TTM_SETDELAYTIME, TTDT_INITIAL, MAKELPARAM(1000, 0)); SendMessage(hTooltip, TTM_SETDELAYTIME, TTDT_RESHOW, MAKELPARAM(-2, 0)); OSVERSIONINFO ov; ov.dwOSVersionInfoSize = sizeof(OSVERSIONINFO); if (::GetVersionEx(&ov) && ov.dwMajorVersion >= 6 && VER_PLATFORM_WIN32_NT == ov.dwPlatformId) { RECT rcMargin; SetRect(&rcMargin, 3, 1, 3, 1); SendMessage(hTooltip, TTM_SETMARGIN, 0, (LPARAM)&rcMargin); } TOOLINFO ti; ZeroMemory(&ti, sizeof(TOOLINFO)); ti.cbSize = sizeof(TOOLINFO); ti.hwnd = hOwner; ti.lpszText = LPSTR_TEXTCALLBACK; SendMessage(hTooltip, TTM_ADDTOOL, 0, (LPARAM)&ti); SendMessage(hTooltip, TTM_SETMAXTIPWIDTH, 0, 1000); return hTooltip; } static void Toolbar_RelayTooltipMsg(HWND hTooltip, HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { MSG msg; msg.hwnd = hwnd; msg.message = uMsg; msg.wParam = wParam; msg.lParam = lParam; //DWORD pts = GetMessagePos(); //POINTSTOPOINT(msg.pt, pts); //msg.time = GetMessageTime(); SendMessage(hTooltip, TTM_RELAYEVENT, 0, (LPARAM)&msg); } static void Toolbar_SetTooltipItem(HWND hwnd, ToolbarItem *item) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; TOOLINFO ti; ti.cbSize = sizeof(TOOLINFO); ti.hwnd = hwnd; ti.uId = 0; ti.lpszText = NULL; if (SendMessage(toolbar->hTooltip, TTM_GETTOOLINFO, 0, (LPARAM)&ti)) { RECT clientRect, toolRect; Toolbar_GetClientRect(hwnd, &clientRect); if (NULL == item || FALSE == item->GetRect(&toolRect)) { SendMessage(toolbar->hTooltip, TTM_ACTIVATE, FALSE, 0L); return; } toolRect.top = clientRect.top; toolRect.bottom = clientRect.bottom; if (ti.lParam != (LPARAM)item || !::EqualRect(&ti.rect, &toolRect)) { CopyRect(&ti.rect, &toolRect); ti.lParam = (LPARAM)item; SendMessage(toolbar->hTooltip, TTM_SETTOOLINFO, 0, (LPARAM)&ti); } SendMessage(toolbar->hTooltip, TTM_ACTIVATE, TRUE, 0L); } } static void Toolbar_ClearBrushCache(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; for (INT i = 0; i < ARRAYSIZE(toolbar->szBrushes); i++) { if (NULL != toolbar->szBrushes[i]) { DeleteObject(toolbar->szBrushes[i]); toolbar->szBrushes[i] = NULL; } } } static HBRUSH Toolbar_GetBrush(HWND hwnd, UINT brushId) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return NULL; if (brushId >= ARRAYSIZE(toolbar->szBrushes)) return NULL; if (NULL == toolbar->szBrushes[brushId]) { COLORREF brushColor = 0x00FF00FF; switch(brushId) { case TOOLBRUSH_BACK: brushColor = toolbar->rgbBk;break; case TOOLBRUSH_FRAME: brushColor = toolbar->rgbFrame; break; case TOOLBRUSH_ITEMBK: brushColor = toolbar->rgbEditBk; break; } toolbar->szBrushes[brushId] = CreateSolidBrush(brushColor); } return toolbar->szBrushes[brushId]; } static HRGN Toolbar_GetFrameRgn(const RECT *prcToolbar, BOOL bottomDock) { if (TOOLBAR_SPACE_TOP < 1) return NULL; HRGN regionFrame; INT origY = (FALSE != bottomDock) ? prcToolbar->top : prcToolbar->bottom - 1; regionFrame = CreateRectRgn(prcToolbar->left, origY, prcToolbar->right, origY + 1); return regionFrame; } static void Toolbar_PaintBack(HWND hwnd, HDC hdc, const RECT *prcToolbar, HRGN regionPaint) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_HIDDEN & toolbar->flags)) { HRGN regionFrame = Toolbar_GetFrameRgn(prcToolbar, (0 != (TBS_BOTTOMDOCK & windowStyle))); if (NULL != regionFrame) { if (FillRgn(hdc, regionFrame, Toolbar_GetBrush(hwnd, TOOLBRUSH_FRAME))) CombineRgn(regionPaint, regionPaint, regionFrame, RGN_DIFF); DeleteObject(regionFrame); } } FillRgn(hdc, regionPaint, Toolbar_GetBrush(hwnd, TOOLBRUSH_BACK)); } static void Toolbar_Paint(HWND hwnd, HDC hdc, const RECT *prcPaint, BOOL fErase) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; RECT clientRect; GetClientRect(hwnd, &clientRect); DWORD windowStyle = GetWindowStyle(hwnd); HRGN paintRegion = CreateRectRgnIndirect(prcPaint); if (0 == (TBS_HIDDEN & toolbar->flags)) { INT savedDC = SaveDC(hdc); HRGN itemRegion = CreateRectRgn(0,0,0,0); SetBkColor(hdc, toolbar->rgbBk); SetTextColor(hdc, toolbar->rgbText); SelectObject(hdc, toolbar->textFont); ToolbarItem *item; RECT paintRect; UINT style; BOOL fFocused = (GetFocus() == hwnd); UINT commonState= 0; if (0 != (WS_DISABLED & windowStyle)) commonState |= ToolbarItem::stateDisabled; size_t count = (NULL != toolbar->items) ? toolbar->items->size() : 0; for (size_t i = 0; i <= count; i++) { item = (i < count) ? toolbar->items->at(i) : toolbar->chevron; if (NULL != item && item->IntersectRect(&paintRect, prcPaint)) { style = item->GetStyle() | commonState; if (i == toolbar->iFocused && fFocused) style |= ToolbarItem::stateFocused; if (0 != (TBS_NOFOCUSRECT & toolbar->flags)) style |= ToolbarItem::stateNoFocusRect; if (0 == (ToolbarItem::stateHidden & style) && item->Paint(hwnd, hdc, &paintRect, style)) { SetRectRgn(itemRegion, paintRect.left, paintRect.top, paintRect.right, paintRect.bottom); CombineRgn(paintRegion, paintRegion, itemRegion, RGN_DIFF); } } } DeleteObject(itemRegion); RestoreDC(hdc, savedDC); } Toolbar_PaintBack(hwnd, hdc, &clientRect, paintRegion); DeleteObject(paintRegion); } static INT Toolbar_FindListItem(ToolbarItem **itemList, INT start, INT count, LPCSTR pszName, INT cchName) { ToolbarItem *item; for (INT i = start; i < count; i++) { item = itemList[i]; if (NULL != item && item->IsEqual(pszName, cchName)) { return i; } } return ITEM_ERR; } static INT Toolbar_ResolveName(HWND hwnd, LPCSTR pszName) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items) return ITEM_ERR; UINT index; UINT count = (UINT)toolbar->items->size(); if (IS_INTRESOURCE(pszName)) { index = (UINT)(UINT_PTR)pszName; if (index == count && NULL != toolbar->chevron) return index; return (index < count) ? index : ITEM_ERR; } INT cchName = lstrlenA(pszName); if (0 == cchName) return ITEM_ERR; if (count == toolbar->resolveCache) { if (toolbar->chevron->IsEqual(pszName, cchName)) index = count; else { index = ITEM_ERR; toolbar->resolveCache = 0; } } else { index = ITEM_ERR; } if (ITEM_ERR == index) index = Toolbar_FindListItem(toolbar->items->size() ? & toolbar->items->at(0) : nullptr, toolbar->resolveCache, count, pszName, cchName); if (ITEM_ERR == index && 0 != toolbar->resolveCache) index = Toolbar_FindListItem(toolbar->items->size() ? &toolbar->items->at(0) : nullptr, 0, toolbar->resolveCache, pszName, cchName); if (ITEM_ERR == index && NULL != toolbar->chevron && toolbar->chevron->IsEqual(pszName, cchName)) index = count; toolbar->resolveCache = (ITEM_ERR != index) ? index : 0; return index; } static BOOL Toolbar_DisplayChevronMenu(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items) return FALSE; HMENU hMenu = CreatePopupMenu(); if (NULL == hMenu) return FALSE; size_t count = toolbar->items->size(); UINT insertedCount = 0; MENUITEMINFO mi; mi.cbSize = sizeof(MENUITEMINFO); UINT style; WCHAR szBuffer[80] = {0}; INT commandId = 0; ToolbarItem *item; BOOL insertBreak = FALSE; for (size_t i = 0; i < count; i++) { item = toolbar->items->at(i); if (NULL != item) { style = item->GetStyle(); if(0 == ((ToolbarItem::stateHidden | ToolbarItem::styleNoChevron) & style) && item->IsRectEmpty() && FALSE != item->FillMenuInfo(hwnd, &mi, szBuffer, ARRAYSIZE(szBuffer))) { if (MIIM_FTYPE == mi.fMask && MFT_MENUBREAK == mi.fType) { if (insertedCount > 0) insertBreak = TRUE; } else { if (FALSE != InsertMenuItem(hMenu, insertedCount, TRUE, &mi)) { if (insertBreak) { mi.fMask = MIIM_FTYPE; mi.fType = MFT_MENUBREAK; if (InsertMenuItem(hMenu, insertedCount, TRUE, &mi)) insertedCount++; insertBreak = FALSE; } insertedCount++; } } } } } if (NULL != hMenu && insertedCount > 0) { RECT windowRect; GetWindowRect(hwnd, &windowRect); UINT windowStyle = GetWindowStyle(hwnd); UINT menuStyle = TPM_RIGHTALIGN | TPM_RETURNCMD; POINT menuOrig; menuOrig.x = windowRect.right; if (0 != (TBS_BOTTOMDOCK & windowStyle)) { menuOrig.y = windowRect.top; menuStyle |= (TPM_BOTTOMALIGN | TPM_VERNEGANIMATION); } else { menuOrig.y = windowRect.bottom; menuStyle |= (TPM_TOPALIGN | TPM_VERPOSANIMATION); } commandId = Menu_TrackPopup(hMenu, menuStyle, menuOrig.x, menuOrig.y, hwnd, NULL); } DestroyMenu(hMenu); if (0 != commandId) Toolbar_SendCommand(hwnd, commandId); return TRUE; } static void Toolbar_DisplayContextMenu(HWND hwnd, INT x, INT y) { HMENU hMenu = Menu_GetMenu(MENU_TOOLBAR, 0); if (NULL != hMenu) { UINT windowStyle = GetWindowStyle(hwnd); CheckMenuRadioItem(hMenu, ID_TOOLBAR_DOCKTOP, ID_TOOLBAR_DOCKBOTTOM, (0 != (TBS_BOTTOMDOCK & windowStyle)) ? ID_TOOLBAR_DOCKBOTTOM :ID_TOOLBAR_DOCKTOP, MF_BYCOMMAND); CheckMenuItem(hMenu, ID_TOOLBAR_AUTOHIDE, MF_BYCOMMAND | ((0 != (TBS_AUTOHIDE & windowStyle)) ? MF_CHECKED :MF_UNCHECKED)); CheckMenuItem(hMenu, ID_TOOLBAR_TABSTOP, MF_BYCOMMAND | ((0 != (TBS_TABSTOP & windowStyle)) ? MF_CHECKED :MF_UNCHECKED)); Menu_TrackPopup(hMenu, TPM_LEFTALIGN | TPM_TOPALIGN, x, y, hwnd, NULL); Menu_ReleaseMenu(hMenu, MENU_TOOLBAR); } } static BOOL Toolbar_InstallMouseHook(HWND hwnd) { if (TLS_OUT_OF_INDEXES == tlsIndex) { tlsIndex = Plugin_TlsAlloc(); if (TLS_OUT_OF_INDEXES == tlsIndex) return FALSE; } TOOLBARMOUSEHOOK *hook = (TOOLBARMOUSEHOOK*)calloc(1, sizeof(TOOLBARMOUSEHOOK)); if (NULL == hook) return FALSE; hook->hwnd = hwnd; hook->hHook = SetWindowsHookEx(WH_MOUSE, Toolbar_MouseHook, NULL, GetCurrentThreadId()); if (NULL == hook->hHook) { free(hook); return FALSE; } Plugin_TlsSetValue(tlsIndex, hook); return TRUE; } static void Toolbar_TrackMouseLeave(HWND hwnd) { TRACKMOUSEEVENT tm; tm.cbSize = sizeof(TRACKMOUSEEVENT); tm.dwFlags = TME_QUERY; tm.hwndTrack = hwnd; if (TrackMouseEvent(&tm) && 0 == (TME_LEAVE & tm.dwFlags)) { tm.cbSize = sizeof(TRACKMOUSEEVENT); tm.dwFlags = TME_LEAVE; tm.hwndTrack = hwnd; TrackMouseEvent(&tm); } } static BOOL Toolbar_RemoveMouseHook(HWND hwnd, BOOL fOnlyMine) { if (TLS_OUT_OF_INDEXES == tlsIndex) return FALSE; TOOLBARMOUSEHOOK *hook = (TOOLBARMOUSEHOOK*)Plugin_TlsGetValue(tlsIndex); if (NULL == hook) return FALSE; if (FALSE != fOnlyMine && hwnd != hook->hwnd) return FALSE; Plugin_TlsSetValue(tlsIndex, NULL); if (NULL != hook->hHook) UnhookWindowsHookEx(hook->hHook); if (NULL != hook->hwnd && hwnd != hook->hwnd) { Toolbar_TrackMouseLeave(hook->hwnd); } free(hook); return TRUE; } static void Toolbar_AutoShowWindow(HWND hwnd) { KillTimer(hwnd, TIMER_AUTOHIDE_ID); TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; HWND hFocus = GetFocus(); BOOL fFocused = (hFocus == hwnd); UINT windowStyle = GetWindowStyle(hwnd); if ((TBS_AUTOHIDE | WS_VISIBLE) != ((TBS_AUTOHIDE | WS_VISIBLE) & windowStyle)) return; if (0 == (TBS_HIDDEN & toolbar->flags)) { if (0 != (TBS_HIDETIMERSET & toolbar->flags)) toolbar->flags &= ~TBS_HIDETIMERSET; if (FALSE == fFocused) Toolbar_TrackMouseLeave(hwnd); return; } Toolbar_RemoveMouseHook(hwnd, FALSE); if (FALSE == fFocused && FALSE == Toolbar_InstallMouseHook(hwnd)) return; toolbar->flags &= ~TBS_HIDDEN; RECT windowRect; if (GetWindowRect(hwnd, &windowRect)) { HWND hParent = GetParent(hwnd); if(NULL != hParent) { MapWindowPoints(HWND_DESKTOP, hParent, (POINT*)&windowRect, 2); INT height = Toolbar_GetIdealHeight(hwnd); INT x = windowRect.left; INT y = windowRect.top; UINT swpFlags = SWP_NOMOVE | SWP_NOREDRAW; UINT animateFlags = AW_SLIDE | AW_VER_POSITIVE; if (0 != (TBS_BOTTOMDOCK & windowStyle)) { y = windowRect.bottom - height; swpFlags &= ~SWP_NOMOVE; animateFlags = AW_SLIDE | AW_VER_NEGATIVE; } SetWindowPos(hwnd, HWND_TOP, x, y, windowRect.right - windowRect.left, height, swpFlags); SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle & ~WS_VISIBLE); BOOL result = AnimateWindow(hwnd, AUTOHIDE_ANIMATETIME, animateFlags); if (!result) { SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle | WS_VISIBLE); InvalidateRect(hwnd, NULL, TRUE); } } } if (FALSE == fFocused) Toolbar_TrackMouseLeave(hwnd); } static void CALLBACK Toolbar_AutoHideTimerProc(HWND hwnd, UINT uMsg, UINT_PTR eventId, DWORD time) { KillTimer(hwnd, eventId); TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_AUTOHIDE & windowStyle)) return; toolbar->flags &= ~TBS_HIDETIMERSET; if (0 != ((TBS_HIDDEN | TBS_MENULOOP) & toolbar->flags)) return; HWND hFocus = ::GetFocus(); if (hwnd == hFocus || IsChild(hwnd, hFocus)) return; Toolbar_RemoveMouseHook(hwnd, TRUE); RECT windowRect; if (GetWindowRect(hwnd, &windowRect)) { HWND hParent = GetParent(hwnd); if(NULL != hParent) { MapWindowPoints(HWND_DESKTOP, hParent, (POINT*)&windowRect, 2); INT height = TOOLBAR_HIDDENHEIGHT; INT x = windowRect.left; INT y = windowRect.top; UINT swpFlags = SWP_NOMOVE | SWP_NOACTIVATE | SWP_NOREDRAW; UINT animateFlags = AW_HIDE | AW_SLIDE | AW_VER_NEGATIVE; HWND zOrder = HWND_TOP; if (0 != (TBS_BOTTOMDOCK & windowStyle)) { y = windowRect.bottom - height; swpFlags &= ~SWP_NOMOVE; zOrder = HWND_BOTTOM; animateFlags = AW_HIDE | AW_SLIDE | AW_VER_POSITIVE; } AnimateWindow(hwnd, AUTOHIDE_ANIMATETIME, animateFlags); if (NULL != toolbar->hBrowser) { RECT invalidRect; CopyRect(&invalidRect, &windowRect); MapWindowPoints(hParent, toolbar->hBrowser, (POINT*)&invalidRect, 2); RedrawWindow(toolbar->hBrowser, &invalidRect, NULL, RDW_ERASE | RDW_INVALIDATE | RDW_ALLCHILDREN); } SetWindowPos(hwnd, zOrder, x, y, windowRect.right - windowRect.left, height, swpFlags); if (0 != (WS_VISIBLE & windowStyle)) { SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle | WS_VISIBLE); InvalidateRect(hwnd, NULL, TRUE); } } } toolbar->flags |= TBS_HIDDEN; } static void Toolbar_AutoHideWindow(HWND hwnd, BOOL fImmediate) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; if (0 != ((TBS_HIDDEN | TBS_MENULOOP) & toolbar->flags)) { return; } UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_AUTOHIDE & windowStyle)) return; if (FALSE == fImmediate) { if (SetTimer(hwnd, TIMER_AUTOHIDE_ID, TIMER_AUTOHIDE_DELAY, Toolbar_AutoHideTimerProc)) toolbar->flags |= TBS_HIDETIMERSET; } else { Toolbar_AutoHideTimerProc(hwnd, WM_TIMER, TIMER_AUTOHIDE_ID, GetTickCount()); } } static void Toolbar_UpdateUiFlags(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; UINT uiState = (UINT)SendMessage(hwnd, WM_QUERYUISTATE, 0, 0L); if (0 != (UISF_HIDEFOCUS & uiState)) toolbar->flags |= TBS_NOFOCUSRECT; else toolbar->flags &= ~TBS_NOFOCUSRECT; } static void Toolbar_NotifyParent(HWND hwnd, UINT eventId) { HWND hParent = GetParent(hwnd); if (NULL == hParent) return; UINT controlId = (UINT)GetWindowLongPtr(hwnd, GWLP_ID); SENDCMD(hParent, controlId, eventId, hwnd); } static LRESULT Toolbar_OnCreate(HWND hwnd, CREATESTRUCT *pcs) { TOOLBAR *toolbar = (TOOLBAR*)calloc(1, sizeof(TOOLBAR)); if (NULL != toolbar) { Toolbar_AddRef(toolbar); SetLastError(ERROR_SUCCESS); if (!SetWindowLongPtr(hwnd, 0, (LONGX86)(LONG_PTR)toolbar) && ERROR_SUCCESS != GetLastError()) { free(toolbar); toolbar = NULL; } } if (NULL == toolbar) { DestroyWindow(hwnd); return -1; } toolbar->chevron = Toolbar_CreateItem(TOOLITEM_CHEVRON, 0); toolbar->items = new ItemList(); toolbar->hTooltip = Toolbar_CreateTooltip(pcs->hInstance, hwnd); if (NULL != toolbar->hTooltip) { ifc_skinhelper *skinHelper; if (SUCCEEDED(Plugin_GetSkinHelper(&skinHelper))) { skinHelper->SkinControl(toolbar->hTooltip, SKINNEDWND_TYPE_TOOLTIP, SWS_USESKINCOLORS | SWS_USESKINCURSORS | SWS_USESKINFONT); skinHelper->Release(); } } if (0 != (TBS_AUTOHIDE & pcs->style)) toolbar->flags |= TBS_HIDDEN; return 0; } static void Toolbar_OnDestroy(HWND hwnd) { Toolbar_RemoveMouseHook(hwnd, TRUE); Toolbar_ClearBrushCache(hwnd); TOOLBAR *toolbar = GetToolbar(hwnd); SetWindowLongPtr(hwnd, 0, 0L); Toolbar_Release(toolbar); } static void Toolbar_OnPaint(HWND hwnd) { PAINTSTRUCT ps; if (BeginPaint(hwnd, &ps)) { if (ps.rcPaint.left != ps.rcPaint.right) Toolbar_Paint(hwnd, ps.hdc, &ps.rcPaint, ps.fErase); EndPaint(hwnd, &ps); } } static void Toolbar_OnPrintClient(HWND hwnd, HDC hdc, UINT options) { RECT clientRect; if (GetClientRect(hwnd, &clientRect)) Toolbar_Paint(hwnd, hdc, &clientRect, 0 != (PRF_ERASEBKGND & options)); } static void Toolbar_OnWindowPosChanged(HWND hwnd, WINDOWPOS *pwp) { if (SWP_NOSIZE == ((SWP_NOSIZE | SWP_FRAMECHANGED) & pwp->flags)) return; Toolbar_UpdateLayout(hwnd, 0 == (SWP_NOREDRAW & pwp->flags)); } static void Toolbar_OnSetRedraw(HWND hwnd, BOOL allowRedraw) { UINT windowStyle = GetWindowStyle(hwnd); if (FALSE == allowRedraw) { if (0 == (TBS_LOCKUPDATE & windowStyle)) SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle | TBS_LOCKUPDATE); } else { if (0 != (TBS_LOCKUPDATE & windowStyle)) { SetWindowLongPtr(hwnd, GWL_STYLE, windowStyle & ~TBS_LOCKUPDATE); Toolbar_UpdateTabstop(hwnd); Toolbar_UpdateLayout(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } } DefWindowProcW(hwnd, WM_SETREDRAW, (WPARAM)allowRedraw, 0L); } static void Toolbar_OnEnterMenuLoop(HWND hwnd, BOOL fContext) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; toolbar->flags |= TBS_MENULOOP; } static void Toolbar_OnExitMenuLoop(HWND hwnd, BOOL fContext) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; toolbar->flags &= ~TBS_MENULOOP; Toolbar_TrackMouseLeave(hwnd); } static void Toolbar_OnSetFocus(HWND hwnd, HWND hFocus) { UINT windowStyle = GetWindowStyle(hwnd); TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; BOOL focusDirLeft = FALSE; if (0 != (0x8000 & GetAsyncKeyState(VK_LEFT))) { focusDirLeft = TRUE; } else if (0 != (0x8000 & GetAsyncKeyState(VK_TAB))) { if (0 != (0x8000 & GetAsyncKeyState(VK_SHIFT))) focusDirLeft = TRUE; } else if (0 != (0x8000 & GetAsyncKeyState(VK_LBUTTON)) || 0 != (0x8000 & GetAsyncKeyState(VK_RBUTTON)) || 0 != (0x8000 & GetAsyncKeyState(VK_MBUTTON)) || 0 != (0x8000 & GetAsyncKeyState(VK_XBUTTON1)) || 0 != (0x8000 & GetAsyncKeyState(VK_XBUTTON2))) { // mouse press (?) } if ((size_t)-1 != toolbar->iFocused) toolbar->iFocused = Toolbar_GetFocusIndex(hwnd, toolbar->iFocused, 0, FALSE); if ((size_t)-1 == toolbar->iFocused) { if (0 != (TBS_TABSTOP & windowStyle)) { if (FALSE == focusDirLeft) toolbar->iFocused = Toolbar_GetFocusIndex(hwnd, 0, 1, FALSE); else { size_t last = (NULL != toolbar->items) ? toolbar->items->size() : 0; if (last > 0) toolbar->iFocused = Toolbar_GetFocusIndex(hwnd, last - 1, -1, FALSE); } } else { size_t lim, index; INT inc; if (FALSE == focusDirLeft) { lim = toolbar->items->size(); index = 0; inc = 1; } else { lim = (size_t)-1; index = toolbar->items->size(); if (index > 0) index--; inc = -1; } for (; index != lim; index += inc) { UINT itemStyle = toolbar->items->at(index)->GetStyle(); UINT mask = ToolbarItem::styleTabstop | ToolbarItem::stateDisabled | ToolbarItem::stateHidden; if (ToolbarItem::styleTabstop == (mask & itemStyle)) { toolbar->iFocused = index; break; } } } } if (((size_t)-1) != toolbar->iFocused) { ToolbarItem *item = Toolbar_GetItem(toolbar, toolbar->iFocused); if (NULL != item) item->SetFocus(hwnd, NULL, TRUE); } if (0 != (WS_TABSTOP & windowStyle)) Toolbar_AutoShowWindow(hwnd); } static void Toolbar_OnKillFocus(HWND hwnd, HWND hFocus) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && ((size_t)-1) != toolbar->iFocused) { ToolbarItem *item = Toolbar_GetItem(toolbar, toolbar->iFocused); if (NULL != item) item->SetFocus(hwnd, NULL, FALSE); toolbar->iFocused = ((size_t)-1); } Toolbar_AutoHideWindow(hwnd, TRUE); } static void Toolbar_OnKeyDown(HWND hwnd, INT vKey, UINT flags) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; Toolbar_AutoShowWindow(hwnd); INT searchDirection = 0; if (VK_LEFT == vKey) searchDirection = -1; else if (VK_RIGHT == vKey) searchDirection = 1; toolbar->iFocused = Toolbar_GetFocusIndex(hwnd, toolbar->iFocused, searchDirection, FALSE); ToolbarItem *focusedItem = Toolbar_GetItem(toolbar, toolbar->iFocused); BOOL fHandled = (NULL != focusedItem && focusedItem->KeyDown(hwnd, vKey, flags)); if (!fHandled) { switch(vKey) { case VK_LEFT: Toolbar_NextItem(hwnd, TBNS_PREVITEM, FALSE); break; case VK_RIGHT: Toolbar_NextItem(hwnd, TBNS_NEXTITEM, FALSE); break; case VK_HOME: case VK_PRIOR: { size_t first = 0; first = Toolbar_GetFocusIndex(hwnd, first, 1, FALSE); if (((size_t)-1) != first) Toolbar_NextItem(hwnd, MAKEINTRESOURCE(first), TRUE); } break; case VK_END: case VK_NEXT: { size_t last = toolbar->items->size(); last = Toolbar_GetFocusIndex(hwnd, last, -1, FALSE); if (((size_t)-1) != last) Toolbar_NextItem(hwnd, MAKEINTRESOURCE(last), TRUE); } break; } } } static void Toolbar_OnKeyUp(HWND hwnd, INT vKey, UINT flags) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; toolbar->iFocused = Toolbar_GetFocusIndex(hwnd, toolbar->iFocused, FALSE, FALSE); ToolbarItem *focusedItem = Toolbar_GetItem(toolbar, toolbar->iFocused); BOOL fHandled = (NULL != focusedItem && focusedItem->KeyUp(hwnd, vKey, flags)); } static void Toolbar_OnMouseMove(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; Toolbar_AutoShowWindow(hwnd); POINT pt; POINTSTOPOINT(pt, pts); RECT clientRect; Toolbar_GetClientRect(hwnd, &clientRect); if (NULL != toolbar->pressed) { if (!PtInRect(&clientRect, pt)) { toolbar->pressed->MouseLeave(hwnd); } else { toolbar->pressed->MouseMove(hwnd, mouseFlags, pt); } return; } ToolbarItem *item = (PtInRect(&clientRect, pt)) ? Toolbar_HitTest(hwnd, pt) : NULL; ToolbarItem *prevHighlighted = toolbar->highlighted; if (NULL != toolbar->highlighted) { if (item != toolbar->highlighted) { toolbar->highlighted->MouseLeave(hwnd); toolbar->highlighted = NULL; } } if (NULL != item) { UINT style = item->GetStyle(); if (0 == ((ToolbarItem::stateDisabled | ToolbarItem::stateHidden | ToolbarItem::styleStatic) & style)) { toolbar->highlighted = item; item->MouseMove(hwnd, mouseFlags, pt); Toolbar_TrackMouseLeave(hwnd); } } if (NULL != toolbar->hTooltip) { if (prevHighlighted == toolbar->highlighted || NULL != prevHighlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_MOUSEMOVE, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); if (prevHighlighted != toolbar->highlighted) { Toolbar_SetTooltipItem(hwnd, toolbar->highlighted); if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_MOUSEMOVE, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); } } } static void Toolbar_OnMouseLeave(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; if (NULL != toolbar->highlighted) { toolbar->highlighted->MouseLeave(hwnd); toolbar->highlighted = NULL; } Toolbar_SetTooltipItem(hwnd, NULL); POINT cursor; if (GetFocus() != hwnd && GetCursorPos(&cursor)) { HWND hCursor = WindowFromPoint(cursor); if (NULL != hCursor && GetWindowThreadProcessId(hwnd, NULL) != GetWindowThreadProcessId(hCursor, NULL)) { Toolbar_AutoHideWindow(hwnd, FALSE); } } } static void Toolbar_OnLButtonDown(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; Toolbar_SetTooltipItem(hwnd, NULL); if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_LBUTTONDOWN, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); POINT pt; POINTSTOPOINT(pt, pts); RECT clientRect; Toolbar_GetClientRect(hwnd, &clientRect); ToolbarItem *item = (PtInRect(&clientRect, pt)) ? Toolbar_HitTest(hwnd, pt) : NULL; if (NULL != item) { if (0 == ((ToolbarItem::stateDisabled | ToolbarItem::stateHidden | ToolbarItem::styleStatic) & item->GetStyle())) { item->LButtonDown(hwnd, mouseFlags, pt); toolbar->pressed = item; } } SetCapture(hwnd); } static void Toolbar_OnLButtonUp(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_LBUTTONUP, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); POINT pt; POINTSTOPOINT(pt, pts); RECT clientRect; Toolbar_GetClientRect(hwnd, &clientRect); ToolbarItem *item = (PtInRect(&clientRect, pt)) ? Toolbar_HitTest(hwnd, pt) : NULL; if (NULL != toolbar->pressed) { ToolbarItem *pressed = toolbar->pressed; toolbar->pressed = NULL; Toolbar_AddRef(toolbar); pressed->AddRef(); if (pressed == item) pressed->Click(hwnd, mouseFlags, pt); pressed->LButtonUp(hwnd, mouseFlags, pt); pressed->Release(); Toolbar_Release(toolbar); } if (hwnd == GetCapture()) ReleaseCapture(); } static void Toolbar_OnRButtonDown(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; Toolbar_SetTooltipItem(hwnd, NULL); if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_RBUTTONDOWN, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); } static void Toolbar_OnRButtonUp(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_RBUTTONUP, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); } static void Toolbar_OnMButtonDown(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; Toolbar_SetTooltipItem(hwnd, NULL); if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_MBUTTONDOWN, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); } static void Toolbar_OnMButtonUp(HWND hwnd, UINT mouseFlags, POINTS pts) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; if (NULL != toolbar->highlighted) Toolbar_RelayTooltipMsg(toolbar->hTooltip, hwnd, WM_MBUTTONUP, (WPARAM)mouseFlags, *((LPARAM*)(&pts))); } static void Toolbar_OnContextMenu(HWND hwnd, HWND hOwner, POINTS pts) { POINT pt; POINTSTOPOINT(pt, pts); if (-1 == pt.x || -1 == pt.y) { RECT windowRect; GetWindowRect(hwnd, &windowRect); if (!GetCursorPos(&pt) || !PtInRect(&windowRect, pt)) { pt.x = windowRect.left + 2; pt.y = windowRect.top + 2; } } else { POINT localPt = pt; MapWindowPoints(HWND_DESKTOP, hwnd, &localPt, 1); RECT clientRect; Toolbar_GetClientRect(hwnd, &clientRect); ToolbarItem *item = (PtInRect(&clientRect, localPt)) ? Toolbar_HitTest(hwnd, localPt) : NULL; if (NULL != item) { UINT itemStyle = item->GetStyle(); if (0 == ((ToolbarItem::stateHidden | ToolbarItem::stateDisabled) & itemStyle)) { RECT itemRect; if (item->GetRect(&itemRect) && PtInRect(&itemRect, localPt)) { if (FALSE != item->DisplayContextMenu(hwnd, pt.x, pt.y)) return; } } } } Toolbar_DisplayContextMenu(hwnd, pt.x, pt.y); } static void Toolbar_OnCaptureChanged(HWND hwnd, HWND hCapture) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; Toolbar_TrackMouseLeave(hwnd); } static LRESULT Toolbar_OnGetFont(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return NULL; return (LRESULT)toolbar->textFont; } static void Toolbar_OnSetFont(HWND hwnd, HFONT hFont, BOOL fRedraw) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; toolbar->textFont = hFont; Toolbar_UpdateTextMetrics(hwnd); if (FALSE != fRedraw) InvalidateRect(hwnd, NULL, TRUE); } static void Toolbar_GetTootipDispInfo(HWND hwnd, NMTTDISPINFO *pdisp) { ToolbarItem *item = (ToolbarItem*)pdisp->lParam; if (NULL != item) { item->GetTip(pdisp->szText, ARRAYSIZE(pdisp->szText)); } } static LRESULT Toolbar_OnTooltipNotify(HWND hwnd, NMHDR *pnmh) { switch(pnmh->code) { case TTN_GETDISPINFO: Toolbar_GetTootipDispInfo(hwnd, (NMTTDISPINFO*)pnmh); break; } return 0; } static LRESULT Toolbar_OnNotify(HWND hwnd, INT controlId, NMHDR *pnmh) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return 0; if (toolbar->hTooltip == pnmh->hwndFrom && NULL != toolbar->hTooltip) return Toolbar_OnTooltipNotify(hwnd, pnmh); return 0; } static void Toolbar_OnCommand(HWND hwnd, INT commandId, INT eventId, HWND hControl) { BOOL fEnabled; switch(commandId) { case ID_TOOLBAR_DOCKTOP: case ID_TOOLBAR_DOCKBOTTOM: fEnabled = (ID_TOOLBAR_DOCKBOTTOM == commandId); if (fEnabled != Toolbar_EnableBottomDock(hwnd, fEnabled)) Toolbar_NotifyParent(hwnd, TBN_DOCKCHANGED); break; case ID_TOOLBAR_AUTOHIDE: fEnabled = (0 == (TBS_AUTOHIDE & GetWindowStyle(hwnd))); if (fEnabled != Toolbar_EnableAutoHide(hwnd, fEnabled)) Toolbar_NotifyParent(hwnd, TBN_AUTOHIDECHANGED); break; case ID_TOOLBAR_TABSTOP: fEnabled = (0 == (TBS_TABSTOP & GetWindowStyle(hwnd))); if (fEnabled != Toolbar_EnableTabStop(hwnd, fEnabled)) Toolbar_NotifyParent(hwnd, TBN_TABSTOPCHANGED); break; } } static BOOL Toolbar_ProcessTabKey(HWND hwnd, UINT messageId) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return FALSE; BOOL focusDirLeft = (0 != (0x8000 & GetAsyncKeyState(VK_SHIFT))); size_t lim, index; INT inc; if (FALSE == focusDirLeft) { inc = 1; lim = toolbar->items->size(); index = toolbar->iFocused; if ((size_t)-1 == index) index = 0; index++; if (index >= lim) return FALSE; } else { inc = -1; lim = (size_t)-1; index = toolbar->iFocused; if ((size_t)-1 == index) index = toolbar->items->size(); index--; if (index >= toolbar->items->size()) return FALSE; } for (; index != lim; index += inc) { ToolbarItem *item = toolbar->items->at(index); if (FALSE != Toolbar_IsItemAcceptFocus(item, TRUE)) { item->SetFocus(hwnd, NULL, TRUE); return TRUE; } } return FALSE; } static LRESULT Toolbar_OnGetDlgCode(HWND hwnd, INT vKey, MSG *pMsg) { if (NULL != pMsg) { switch(vKey) { case VK_TAB: if (FALSE == Toolbar_ProcessTabKey(hwnd, pMsg->message)) { if (WM_KEYDOWN == pMsg->message) Toolbar_AutoHideWindow(hwnd, TRUE); return 0; } break; } } return DLGC_WANTALLKEYS; } static void Toolbar_OnUpdateUiState(HWND hwnd, INT action, INT state) { DefWindowProc(hwnd, WM_UPDATEUISTATE, MAKEWPARAM(action, state), 0L); Toolbar_UpdateUiFlags(hwnd); } static void Toolbar_OnEnable(HWND hwnd, BOOL fEnable) { InvalidateRect(hwnd, NULL, FALSE); } static LRESULT Toolbar_OnSetCursor(HWND hwnd, HWND hCursor, UINT hitTest, UINT messageId) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return FALSE; if (0 == (TBS_MENULOOP & toolbar->flags) && hwnd == hCursor) { ToolbarItem *item = (NULL != toolbar->pressed) ? toolbar->pressed : toolbar->highlighted; if (NULL != item && FALSE != item->SetCursor(hwnd, hCursor, hitTest, messageId)) { return TRUE; } } return DefWindowProcW(hwnd, WM_SETCURSOR, (WPARAM)hCursor, MAKELPARAM(hitTest, messageId)); } static LRESULT Toolbar_OnColorEdit(HWND hwnd, HDC hdcCtrl, HWND hwndCtrl) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return 0; SetTextColor(hdcCtrl, toolbar->rgbEdit); SetBkColor(hdcCtrl, toolbar->rgbEditBk); return (LRESULT)Toolbar_GetBrush(hwnd, TOOLBRUSH_ITEMBK); } static LRESULT Toolbar_OnColorStatic(HWND hwnd, HDC hdcCtrl, HWND hwndCtrl) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return 0; SetTextColor(hdcCtrl, toolbar->rgbText); SetBkColor(hdcCtrl, toolbar->rgbBk); return (LRESULT)Toolbar_GetBrush(hwnd, TOOLBRUSH_BACK); } static LRESULT Toolbar_OnGetIdealHeight(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->imageList) return 0; INT toolbarHeight, iconWidth; if (!ImageList_GetIconSize(toolbar->imageList, &iconWidth, &toolbarHeight)) toolbarHeight = 0; else toolbarHeight /= TOOLBAR_ICONSTATE_COUNT; HDC hdc = GetDCEx(hwnd, NULL, DCX_CACHE | DCX_NORESETATTRS); if (NULL != hdc) { HFONT originalFont = (HFONT)SelectObject(hdc, toolbar->textFont); TEXTMETRIC tm; if (GetTextMetrics(hdc, &tm)) { tm.tmHeight += ((tm.tmHeight - (tm.tmAscent - tm.tmInternalLeading))/2 + 2); if (toolbarHeight <= tm.tmHeight) toolbarHeight = tm.tmHeight + 4; } SelectObject(hdc, originalFont); ReleaseDC(hwnd, hdc); } return toolbarHeight + TOOLBAR_SPACE_TOP + TOOLBAR_SPACE_BOTTOM; } static void Toolbar_OnUpdateSkin(HWND hwnd, BOOL fRedraw) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return; ifc_skinhelper *skinHelper; if (FAILED(Plugin_GetSkinHelper(&skinHelper))) skinHelper = NULL; Toolbar_UpdateColorTable(hwnd, skinHelper); Toolbar_ClearBrushCache(hwnd); if (NULL != toolbar->imageList) ImageList_Destroy(toolbar->imageList); toolbar->imageList = Toolbar_LoadImagelist(toolbar, Plugin_GetInstance(), MAKEINTRESOURCE(IDR_TOOLBARLARGE_IMAGE), 21, TOOLBAR_ICONSTATE_COUNT); if (NULL != toolbar->ownedFont) { DeleteObject(toolbar->ownedFont); } HFONT font = (NULL != skinHelper) ? skinHelper->GetFont() : NULL; if (0/* create custom font */) { toolbar->ownedFont = Toolbar_CreateFont(font); if (NULL != toolbar->ownedFont) font = toolbar->ownedFont; } else { toolbar->ownedFont = NULL; } SendMessage(hwnd, WM_SETFONT, (WPARAM)font, FALSE); if (NULL != toolbar->hTooltip) { SendMessage(toolbar->hTooltip, WM_SETFONT, (WPARAM)(toolbar->textFont), TRUE); } Toolbar_UpdateUiFlags(hwnd); if (NULL != toolbar->items) { size_t index = toolbar->items->size(); while(index-- > 0) { ToolbarItem *item = toolbar->items->at(index); if (NULL != item) { item->UpdateSkin(hwnd); } } } Toolbar_UpdateLayout(hwnd, FALSE); if (FALSE != fRedraw) InvalidateRect(hwnd, NULL, TRUE); if (NULL != skinHelper) skinHelper->Release(); } static LRESULT Toolbar_OnGetIconSize(HWND hwnd, INT iconIndex, SIZE *pSize) { if (NULL == pSize) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (ICON_CHEVRON == iconIndex) { UINT windowStyle = GetWindowStyle(hwnd); iconIndex = (0 != (TBS_BOTTOMDOCK & windowStyle)) ? ICON_CHEVRON_BOTTOM : ICON_CHEVRON_TOP; } if (NULL == toolbar || NULL == toolbar->imageList || iconIndex < 0 || iconIndex >= ImageList_GetImageCount(toolbar->imageList)) { ZeroMemory(pSize, sizeof(SIZE)); return FALSE; } INT cx, cy; BOOL result = ImageList_GetIconSize(toolbar->imageList, &cx, &cy); if (FALSE != result) { pSize->cy = cy / TOOLBAR_ICONSTATE_COUNT; switch(iconIndex) { case ICON_SEPARATOR: pSize->cx = ICON_SEPARATOR_WIDTH; break; case ICON_CHEVRON_TOP: case ICON_CHEVRON_BOTTOM: pSize->cx = ICON_CHEVRON_WIDTH; break; case ICON_HISTORY: pSize->cx = ICON_HISTORY_WIDTH; break; case ICON_BACK: case ICON_FORWARD: pSize->cx =ICON_ARROW_WIDTH; break; default: pSize->cx = cx; break; } } else ZeroMemory(pSize, sizeof(SIZE)); return result; } static LRESULT Toolbar_OnSendCommand(HWND hwnd, INT commandId) { HWND hParent = GetParent(hwnd); if (NULL == hParent) return FALSE; if (ID_CHEVRON_CLICKED == commandId) { Toolbar_DisplayChevronMenu(hwnd); return 0; } TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->items) { size_t index= toolbar->items->size(); while(index--) toolbar->items->at(index)->CommandSent(hwnd, commandId); } return SendMessage(hParent, WM_COMMAND, MAKEWPARAM(commandId, 0), (LPARAM)hwnd); } static LRESULT Toolbar_OnDrawIcon(HWND hwnd, TOOLBARDRAWICONPARAM *iconParam) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == iconParam) return FALSE; INT iconIndex = iconParam->iconIndex; if (ICON_CHEVRON == iconIndex) { UINT windowStyle = GetWindowStyle(hwnd); iconIndex = (0 != (TBS_BOTTOMDOCK & windowStyle)) ? ICON_CHEVRON_BOTTOM : ICON_CHEVRON_TOP; } if (NULL == toolbar->imageList || iconIndex < 0 || iconIndex >= ImageList_GetImageCount(toolbar->imageList)) { return FALSE; } INT frameHeight, frameWidth; if (!ImageList_GetIconSize(toolbar->imageList, &frameWidth, &frameHeight)) { frameHeight = 0; frameWidth = 0; } INT offsetY; if (0 != (ToolbarItem::stateDisabled & iconParam->itemState)) offsetY = TOOLBAR_ICONSTATE_DISABLED; else if (0 != (ToolbarItem::statePressed & iconParam->itemState)) offsetY = TOOLBAR_ICONSTATE_PRESSED; else if (0 != (ToolbarItem::stateHighlighted & iconParam->itemState)) offsetY = TOOLBAR_ICONSTATE_HIGHLIGHTED; else offsetY = TOOLBAR_ICONSTATE_NORMAL; frameHeight = frameHeight / TOOLBAR_ICONSTATE_COUNT; IMAGELISTDRAWPARAMS param; param.cbSize = sizeof(IMAGELISTDRAWPARAMS) - sizeof(DWORD) * 3; param.himl = toolbar->imageList; param.i = iconIndex; param.hdcDst = iconParam->hdcDst; param.x = iconParam->x; param.y = iconParam->y; param.cx = (iconParam->cx > frameWidth) ? frameWidth : iconParam->cx; param.cy = (iconParam->cy > frameHeight) ? frameHeight : iconParam->cy; param.xBitmap = 0; param.yBitmap = frameHeight * offsetY; param.rgbBk = CLR_NONE; param.rgbFg = CLR_NONE; param.fStyle = ILD_NORMAL; param.dwRop = SRCCOPY; param.fState = ILS_NORMAL; param.Frame = 0; param.crEffect = 0; return ImageList_DrawIndirect(¶m); } static LRESULT Toolbar_OnGetItemCount(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items) return 0; return (INT)toolbar->items->size(); } static LRESULT Toolbar_OnClear(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return FALSE; if (NULL == toolbar->items) return TRUE; Toolbar_ClearItems(toolbar); UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_LOCKUPDATE & windowStyle)) { Toolbar_UpdateTabstop(hwnd); Toolbar_UpdateLayout(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } return TRUE; } static LRESULT Toolbar_OnInsertItem(HWND hwnd, TOOLBARINSERTITEM *insertItem) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items || NULL == insertItem || insertItem->cbSize != sizeof(TOOLBARINSERTITEM)) { return -1; } INT index = ITEM_ERR; UINT styleOverride = 0; if (0 != (TBIS_HIDDEN & insertItem->style)) styleOverride |= ToolbarItem::stateHidden; if (0 != (TBIS_DISABLED & insertItem->style)) styleOverride |= ToolbarItem::stateDisabled; if (0 != (TBIS_CHEVRONONLY & insertItem->style)) styleOverride |= ToolbarItem::styleChevronOnly; if (0 != (TBIS_NOCHEVRON & insertItem->style)) styleOverride |= ToolbarItem::styleNoChevron; if (0 != (TBIS_POPUP & insertItem->style)) styleOverride |= ToolbarItem::stylePopup; ToolbarItem *item = Toolbar_CreateItem(insertItem->pszName, styleOverride); if (NULL != item) { index = Toolbar_InsertItemInternal(toolbar, item, insertItem->insertBefore, hwnd); item->Release(); if (ITEM_ERR != index) { UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_LOCKUPDATE & windowStyle)) { Toolbar_UpdateTabstop(hwnd); Toolbar_UpdateLayout(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } } } return index; } static LRESULT Toolbar_OnFindItem(HWND hwnd, LPCSTR pszName) { return Toolbar_ResolveName(hwnd, pszName); } static LRESULT Toolbar_OnRemoveItem(HWND hwnd, LPCSTR pszName) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items || index == toolbar->items->size()) return FALSE; ToolbarItem *item = toolbar->items->at(index); toolbar->items->erase(toolbar->items->begin() + index); if (NULL != item) item->Release(); UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_LOCKUPDATE & windowStyle)) { Toolbar_UpdateTabstop(hwnd); Toolbar_UpdateLayout(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } return TRUE; } static LRESULT Toolbar_OnSetItemInt(HWND hwnd, LPCSTR pszName, INT value) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return FALSE; return toolbar->items->at(index)->SetValueInt(hwnd, value); } static LRESULT Toolbar_OnSetItemString(HWND hwnd, LPCSTR pszName, LPCWSTR value) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return FALSE; return toolbar->items->at(index)->SetValueStr(hwnd, value); } static COLORREF Toolbar_OnGetBkColor(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); return (NULL != toolbar) ? toolbar->rgbBk : 0x00FF00FF; } static COLORREF Toolbar_OnGetFgColor(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); return (NULL != toolbar) ? toolbar->rgbFg : 0x00FF00FF; } static COLORREF Toolbar_OnGetTextColor(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); return (NULL != toolbar) ? toolbar->rgbText : 0x00FF00FF; } static COLORREF Toolbar_OnGetHiliteColor(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); return (NULL != toolbar) ? toolbar->rgbHilite : 0x00FF00FF; } static COLORREF Toolbar_OnGetEditColor(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); return (NULL != toolbar) ? toolbar->rgbEdit : 0x00FF00FF; } static COLORREF Toolbar_OnGetEditBkColor(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); return (NULL != toolbar) ? toolbar->rgbEditBk : 0x00FF00FF; } static LRESULT Toolbar_OnEnableItem(HWND hwnd, LPCSTR pszName, BOOL fEnable) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return FALSE; ToolbarItem *item = toolbar->items->at(index); if (NULL == item) return FALSE; item->SetStyle(hwnd, (0 != fEnable) ? 0 : ToolbarItem::stateDisabled, ToolbarItem::stateDisabled); if (FALSE == fEnable && toolbar->highlighted == item) Toolbar_SetTooltipItem(hwnd, NULL); Toolbar_UpdateTabstop(hwnd); return TRUE; } static LRESULT Toolbar_OnGetItemStyle(HWND hwnd, LPCSTR pszName, UINT fMask) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return 0; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return 0; UINT itemStyle = toolbar->items->at(index)->GetStyle(); return (itemStyle & fMask); } static LRESULT Toolbar_OnGetItemCommand(HWND hwnd, LPCSTR pszName) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return 0; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return 0; return toolbar->items->at(index)->GetCommandId(); } static LRESULT Toolbar_OnShowItem(HWND hwnd, LPCSTR pszName, BOOL fShow) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return FALSE; ToolbarItem *item = toolbar->items->at(index); if (NULL == item) return FALSE; UINT style = item->GetStyle(); if ((0 != (ToolbarItem::stateHidden & style)) == (FALSE == fShow)) return TRUE; item->SetStyle(hwnd, (0 != fShow) ? 0 : ToolbarItem::stateHidden, ToolbarItem::stateHidden); UINT windowStyle = GetWindowStyle(hwnd); if (0 == (TBS_LOCKUPDATE & windowStyle)) { Toolbar_UpdateTabstop(hwnd); Toolbar_UpdateLayout(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } return TRUE; } static LRESULT Toolbar_OnSetItemDescription(HWND hwnd, LPCSTR pszName, LPCWSTR pszDescription) { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (index == toolbar->items->size()) return FALSE; ToolbarItem *item = toolbar->items->at(index); if (NULL == item) return FALSE; if (FALSE == item->SetDescription(hwnd, pszDescription)) return FALSE; if (IsWindowVisible(hwnd) && IsWindowEnabled(hwnd)) Toolbar_UpdateTip(hwnd); return TRUE; } static void Toolbar_OnUpdateTip(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->hTooltip) { TOOLINFO ti; ti.cbSize = sizeof(TOOLINFO); ti.uId = 0; ti.hinst = NULL; ti.hwnd = hwnd; ti.lpszText = LPSTR_TEXTCALLBACK; SendMessage(toolbar->hTooltip, TTM_UPDATETIPTEXT, 0, (LPARAM)&ti); } } static LRESULT Toolbar_OnGetTextMetrics(HWND hwnd, TOOLBARTEXTMETRIC *metric) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return FALSE; CopyMemory(metric, &toolbar->textMetric, sizeof(TOOLBARTEXTMETRIC)); return TRUE; } static LRESULT Toolbar_OnGetBkBrush(HWND hwnd) { return (LRESULT)Toolbar_GetBrush(hwnd, TOOLBRUSH_BACK); } static LRESULT Toolbar_OnLayout(HWND hwnd, TOOLBARLAYOUT *layout) { if (NULL == layout) return FALSE; CopyRect(&layout->clientRect, layout->prcParent); DWORD windowStyle = GetWindowStyle(hwnd); TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || 0 == (WS_VISIBLE & windowStyle)) { layout->insertAfter = HWND_BOTTOM; SetRectEmpty(&layout->toolbarRect); return TRUE; } CopyRect(&layout->toolbarRect, layout->prcParent); INT toolbarHeight, clientInflate; if (0 != (TBS_AUTOHIDE & windowStyle)) { clientInflate = TOOLBAR_HIDDENHEIGHT; toolbarHeight = (0 == (TBS_HIDDEN & toolbar->flags)) ? Toolbar_GetIdealHeight(hwnd) : TOOLBAR_HIDDENHEIGHT; } else { toolbarHeight = Toolbar_GetIdealHeight(hwnd); clientInflate = toolbarHeight; } if (0 != (TBS_BOTTOMDOCK & windowStyle)) { layout->toolbarRect.top = layout->toolbarRect.bottom - toolbarHeight; if (layout->toolbarRect.top < layout->prcParent->top) layout->toolbarRect.top = layout->prcParent->top; layout->clientRect.bottom -= clientInflate; layout->insertAfter = (0 == (TBS_AUTOHIDE & windowStyle) || 0 != (TBS_HIDDEN & toolbar->flags)) ? HWND_BOTTOM : HWND_TOP; } else { layout->toolbarRect.bottom = layout->toolbarRect.top + toolbarHeight; if (layout->toolbarRect.bottom > layout->prcParent->bottom) layout->toolbarRect.bottom = layout->prcParent->bottom; layout->clientRect.top += clientInflate; layout->insertAfter = HWND_TOP; } return TRUE; } static BOOL Toolbar_OnNextItem(HWND hwnd, LPCSTR pszName, BOOL fUseName) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return FALSE; size_t focusIndex = toolbar->iFocused; if (FALSE == fUseName) { focusIndex = Toolbar_GetFocusIndex(hwnd, toolbar->iFocused, 0, FALSE); if (((size_t)-1) == focusIndex) { if (TBNS_NEXTITEM == pszName) { focusIndex = toolbar->items->size(); if (focusIndex > 0) focusIndex--; } else { focusIndex = 0; } } else { if (TBNS_NEXTITEM == pszName) { if (focusIndex <= toolbar->items->size()) focusIndex = Toolbar_GetFocusIndex(hwnd, focusIndex + 1, 1, FALSE); } else { if (focusIndex > 0) focusIndex = Toolbar_GetFocusIndex(hwnd, focusIndex - 1, -1, FALSE); } if (((size_t)-1) == focusIndex) focusIndex = toolbar->iFocused; } } else { INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index || !Toolbar_IsItemAcceptFocus(Toolbar_GetItem(toolbar, index), FALSE)) return FALSE; focusIndex = (size_t)index; } BOOL focusChanged = FALSE; if (focusIndex != toolbar->iFocused) { ToolbarItem *itemNew, *itemOld; itemOld = Toolbar_GetItem(toolbar, toolbar->iFocused); itemNew = Toolbar_GetItem(toolbar, focusIndex); toolbar->iFocused = focusIndex; if (NULL != itemOld) itemOld->SetFocus(hwnd, itemNew, FALSE); if (NULL != itemNew) itemNew->SetFocus(hwnd, itemOld, TRUE); focusChanged = TRUE; } HWND hFocus = GetFocus(); if (hwnd != hFocus && FALSE == IsChild(hwnd, hFocus)) { if (IsWindowVisible(hwnd) && IsWindowEnabled(hwnd)) { HWND hRoot = GetAncestor(hwnd, GA_ROOT); if (NULL != hRoot) SendMessage(hwnd, WM_NEXTDLGCTL, (WPARAM)hwnd, TRUE); if (hwnd != GetFocus()) SetFocus(hwnd); } } else if (FALSE == focusChanged) { ToolbarItem *item = Toolbar_GetItem(toolbar, toolbar->iFocused); if (NULL != item) item->SetFocus(hwnd, item, TRUE); } return TRUE; } static BOOL Toolbar_OnGetItemInfo(HWND hwnd, LPCSTR pszName, TBITEMINFO *itemInfo) { if (NULL == itemInfo) return FALSE; INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || index == toolbar->items->size()) return FALSE; ToolbarItem *item = toolbar->items->at(index); if (NULL == item) return FALSE; BOOL result = TRUE; itemInfo->commandId = item->GetCommandId(); itemInfo->style = item->GetStyle(); if (NULL != itemInfo->pszText) { if (FAILED(item->GetText(itemInfo->pszText, itemInfo->cchText))) result = FALSE; } if (NULL != itemInfo->pszDescription) { if (FAILED(item->GetDescription(itemInfo->pszDescription, itemInfo->cchDescription))) result = FALSE; } return result; } static HRESULT Toolbar_GetServiceCommandState(HWND hBrowser,ifc_omservicecommand *serviceCommand, const GUID *commandGroup, UINT commandId, HRESULT defaultState) { HRESULT state = (NULL != serviceCommand) ? serviceCommand->QueryState(hBrowser, commandGroup, commandId) : defaultState; if (E_NOTIMPL == state) state = defaultState; return state; } static INT Toolbar_InsertItemHelper(HWND hwnd, LPCSTR pszName, UINT overrideStyle, INT insertBefore) { INT index = ITEM_ERR; TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->items) { ToolbarItem *item = Toolbar_CreateItem(pszName, overrideStyle); if (NULL != item) { index = Toolbar_InsertItemInternal(toolbar, item, insertBefore, hwnd); item->Release(); } } return index; } static INT Toolbar_AddItemHelper(HWND hwnd, LPCSTR pszName, UINT overrideStyle) { return Toolbar_InsertItemHelper(hwnd, pszName, overrideStyle, TBIP_LAST); } static INT Toolbar_AddItemHelper2(HWND hwnd, LPCSTR pszName, UINT overrideStyle, ifc_omservicecommand *serviceCommand, const GUID *commandGroup, ULONG commandId, HRESULT defaultState) { HWND hBrowser = GetParent(hwnd); HRESULT commandState = Toolbar_GetServiceCommandState(hBrowser, serviceCommand, commandGroup, commandId, defaultState); if (CMDSTATE_ENABLED != commandState) return ITEM_ERR; return Toolbar_AddItemHelper(hwnd, pszName, overrideStyle); } static LRESULT Toolbar_OnAutoPopulate(HWND hwnd, ifc_omservice *service, UINT flags) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || NULL == toolbar->items) return 0; ifc_omservicecommand *serviceCommand; if (NULL == service || FAILED(service->QueryInterface(IFC_OmServiceCommand, (void**)&serviceCommand))) serviceCommand = NULL; UINT windowStyle = GetWindowStyle(hwnd); Toolbar_ClearItems(toolbar); ToolbarItem *item; HRESULT commandStatus; size_t blockSize, currentSize; HWND hBrowser = GetParent(hwnd); // Back/Forward commandStatus = Toolbar_GetServiceCommandState(hBrowser, serviceCommand, &CMDGROUP_NAVIGATION, NAVCOMMAND_BACKFORWARD, CMDSTATE_ENABLED); if (SUCCEEDED(commandStatus)) { Toolbar_AddItemHelper(hwnd, TOOLITEM_BUTTON_BACK, ToolbarItem::stateDisabled); Toolbar_AddItemHelper(hwnd, TOOLITEM_BUTTON_FORWARD, ToolbarItem::stateDisabled); // History if (0 != toolbar->items->size()) { Toolbar_AddItemHelper2(hwnd, TOOLITEM_BUTTON_HISTORY, ToolbarItem::stateDisabled | ToolbarItem::styleNoChevron, serviceCommand, &CMDGROUP_NAVIGATION, NAVCOMMAND_HISTORY, CMDSTATE_ENABLED); } } blockSize = toolbar->items->size(); // Home Toolbar_AddItemHelper2(hwnd, TOOLITEM_BUTTON_HOME, ToolbarItem::stateDisabled, serviceCommand, &CMDGROUP_NAVIGATION, NAVCOMMAND_HOME, CMDSTATE_ENABLED); // Refresh Toolbar_AddItemHelper2(hwnd, TOOLITEM_BUTTON_REFRESH, ToolbarItem::stateDisabled, serviceCommand, &CMDGROUP_NAVIGATION, NAVCOMMAND_REFRESH, CMDSTATE_ENABLED); // Stop Toolbar_AddItemHelper2(hwnd, TOOLITEM_BUTTON_STOP, ToolbarItem::stateDisabled, serviceCommand, &CMDGROUP_NAVIGATION, NAVCOMMAND_STOP, CMDSTATE_ENABLED); currentSize = toolbar->items->size(); if (0 != blockSize && 0 != currentSize && blockSize != currentSize) Toolbar_InsertItemHelper(hwnd, TOOLITEM_SPACE, ToolbarItem::styleNoChevron, (INT)blockSize); BOOL fShowFlexSpace = TRUE; // Addressbar UINT style = ToolbarItem::styleNoChevron | ToolbarItem::stateDisabled | ToolbarItem::stylePopup; if (0 == (TBS_FORCEADDRESS & windowStyle) && CMDSTATE_ENABLED != Toolbar_GetServiceCommandState(hBrowser, serviceCommand, &CMDGROUP_ADDRESSBAR, ADDRESSCOMMAND_VISIBLE, (0 != (TBS_SHOWADDRESS & windowStyle)) ? CMDSTATE_ENABLED : CMDSTATE_DISABLED)) { style |= ToolbarItem::stateHidden; } if (CMDSTATE_ENABLED == Toolbar_GetServiceCommandState(hBrowser, serviceCommand, &CMDGROUP_ADDRESSBAR, ADDRESSCOMMAND_READONLY, (0 != (TBPF_READONLYADDRESS & flags) ? CMDSTATE_ENABLED : CMDSTATE_DISABLED))) { style |= ToolbarAddress::styleAddressReadonly; } if (0 == (TBS_FANCYADDRESS & windowStyle)) style |= ToolbarAddress::styleAddressShowReal; if (ITEM_ERR != Toolbar_AddItemHelper(hwnd, TOOLITEM_ADDRESSBAR, style)) { currentSize++; if (0 == (ToolbarItem::stateHidden & style)) fShowFlexSpace = FALSE; } if (0 != currentSize) { Toolbar_AddItemHelper(hwnd, TOOLITEM_FLEXSPACE, ToolbarItem::stylePopup | ((FALSE == fShowFlexSpace) ? ToolbarItem::stateHidden : 0)); } if (NULL != serviceCommand && 0 == (TBPF_NOSERVICECOMMANDS & flags)) { blockSize = toolbar->items->size(); // Rating commandStatus = Toolbar_GetServiceCommandState(hBrowser, serviceCommand, &CMDGROUP_SERVICE, SVCCOMMAND_RATE, CMDSTATE_UNKNOWN); if (SUCCEEDED(commandStatus)) { UINT rating; if (NULL != service && SUCCEEDED(service->GetRating(&rating))) { if (ITEM_ERR != Toolbar_AddItemHelper(hwnd, TOOLITEM_USERRATING, 0)) Toolbar_SetItemInt(hwnd, TOOLITEM_USERRATING, rating); } } // Show Info Toolbar_AddItemHelper2(hwnd, TOOLITEM_CMDLINK_INFO, ToolbarItem::styleChevronOnly, serviceCommand, &CMDGROUP_SERVICE, SVCCOMMAND_SHOWINFO, CMDSTATE_UNKNOWN); // Report Toolbar_AddItemHelper2(hwnd, TOOLITEM_CMDLINK_REPORT, ToolbarItem::styleChevronOnly, serviceCommand, &CMDGROUP_SERVICE, SVCCOMMAND_REPORT, CMDSTATE_UNKNOWN); // Remove Toolbar_AddItemHelper2(hwnd, TOOLITEM_CMDLINK_UNSUBSCRIBE, ToolbarItem::styleChevronOnly, serviceCommand, &CMDGROUP_SERVICE, SVCCOMMAND_UNSUBSCRIBE, CMDSTATE_UNKNOWN); currentSize = toolbar->items->size(); if (0 != currentSize && blockSize != currentSize) Toolbar_AddItemHelper(hwnd, TOOLITEM_SEPARATOR, ToolbarItem::styleChevronOnly); } const TOOLBARITEMINSERTREC szBrowserTools[] = { { TOOLITEM_BUTTON_SECURECONNECTION, ToolbarItem::stateHidden | ToolbarItem::stylePopup | ToolbarItem::styleNoChevron /*| TBIS_NOREMOVE*/ }, { TOOLITEM_BUTTON_SCRIPTERROR, ToolbarItem::stateHidden | ToolbarItem::stylePopup | ToolbarItem::styleNoChevron /*| TBIS_NOREMOVE*/}, { TOOLITEM_DOWNLOADPROGRESS, ToolbarItem::styleNoChevron | ToolbarItem::stateDisabled /*| TBIS_NOREMOVE*/}, }; for (INT i = 0; i < ARRAYSIZE(szBrowserTools); i++) { item = Toolbar_CreateItem(szBrowserTools[i].name, szBrowserTools[i].style); if (NULL != item) { Toolbar_InsertItemInternal(toolbar, item, TBIP_LAST, hwnd); item->Release(); } } if (0 == (TBS_LOCKUPDATE & windowStyle)) { Toolbar_UpdateTabstop(hwnd); Toolbar_UpdateLayout(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } if (NULL != serviceCommand) serviceCommand->Release(); return (INT)(INT_PTR)toolbar->items->size(); } static LRESULT Toolbar_OnEnableBottomDock(HWND hwnd, BOOL fEnable) { UINT windowStyle = GetWindowStyle(hwnd); UINT newStyle = windowStyle; if (FALSE == fEnable) newStyle &= ~TBS_BOTTOMDOCK; else newStyle |= TBS_BOTTOMDOCK; if(newStyle == windowStyle) return fEnable; if (0 != (TBS_AUTOHIDE & windowStyle)) { HWND hFocus = GetFocus(); if (hwnd == hFocus || IsChild(hwnd, hFocus)) Toolbar_AutoHideWindow(hwnd, TRUE); } SetWindowLongPtr(hwnd, GWL_STYLE, newStyle); HWND hParent = GetParent(hwnd); if (NULL != hParent) { SetWindowPos(hParent, NULL, 0, 0, 0, 0, SWP_NOSIZE |SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE | SWP_FRAMECHANGED); } HWND insertAfter = HWND_TOP; if (0 != (TBS_BOTTOMDOCK & newStyle)) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar || 0 == (TBS_AUTOHIDE & windowStyle) || 0 != (TBS_HIDDEN & toolbar->flags)) { insertAfter = HWND_BOTTOM; } } SetWindowPos(hwnd, insertAfter, 0, 0, 0, 0, SWP_NOACTIVATE | SWP_NOSIZE | SWP_NOMOVE | SWP_NOREDRAW); InvalidateRect(hwnd, NULL, TRUE); return !fEnable; } static LRESULT Toolbar_OnEnableAutoHide(HWND hwnd, BOOL fEnable) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return 0; UINT windowStyle = GetWindowStyle(hwnd); UINT newStyle = windowStyle; KillTimer(hwnd, TIMER_AUTOHIDE_ID); toolbar->flags &= ~TBS_HIDETIMERSET; if (FALSE == fEnable) { newStyle &= ~TBS_AUTOHIDE; toolbar->flags &= ~TBS_HIDDEN; } else { newStyle |= TBS_AUTOHIDE; HWND hFocus = GetFocus(); if (hwnd != hFocus && FALSE == IsChild(hwnd, hFocus)) toolbar->flags |= TBS_HIDDEN; } if(newStyle == windowStyle) return fEnable; SetWindowLongPtr(hwnd, GWL_STYLE, newStyle); HWND hParent = GetParent(hwnd); if (NULL != hParent) { SetWindowPos(hParent, NULL, 0, 0, 0, 0, SWP_NOSIZE |SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE | SWP_FRAMECHANGED); } InvalidateRect(hwnd, NULL, TRUE); return !fEnable; } static LRESULT Toolbar_OnEnableTabStop(HWND hwnd, BOOL fEnable) { UINT windowStyle = GetWindowStyle(hwnd); UINT newStyle = windowStyle; if (FALSE == fEnable) newStyle &= ~TBS_TABSTOP; else newStyle |= TBS_TABSTOP; if(newStyle == windowStyle) return fEnable; SetWindowLongPtr(hwnd, GWL_STYLE, newStyle); Toolbar_UpdateTabstop(hwnd); return !fEnable; } static LRESULT Toolbar_OnEnableForceAddress(HWND hwnd, BOOL fEnable) { UINT windowStyle = GetWindowStyle(hwnd); UINT newStyle = windowStyle; if (FALSE == fEnable) newStyle &= ~TBS_FORCEADDRESS; else newStyle |= TBS_FORCEADDRESS; if(newStyle == windowStyle) return fEnable; SetWindowLongPtr(hwnd, GWL_STYLE, newStyle); TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->items) { size_t index = toolbar->items->size(); while(index--) { ToolbarItem *item = toolbar->items->at(index); if (NULL != item && item->IsEqual(TOOLITEM_ADDRESSBAR, -1)) { UINT itemStyle = item->GetStyle(); if (FALSE != fEnable) { if (0 != (ToolbarItem::stateHidden & itemStyle)) { Toolbar_LockUpdate(hwnd, TRUE); if (FALSE != Toolbar_ShowItem(hwnd, MAKEINTRESOURCE(index), TRUE)) { size_t nextIndex = index + 1; if (nextIndex < toolbar->items->size()) { ToolbarItem *nextItem = toolbar->items->at(nextIndex); if (NULL != nextItem && nextItem->IsEqual(TOOLITEM_FLEXSPACE, -1)) { Toolbar_ShowItem(hwnd, MAKEINTRESOURCE(nextIndex), FALSE); } } } Toolbar_LockUpdate(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } } else { HRESULT commandState = E_NOTIMPL; HWND hBrowser = GetParent(hwnd); ifc_omservice *service; if (NULL != hBrowser && FALSE != BrowserControl_GetService(hBrowser, &service)) { ifc_omservicecommand *serviceCommand; if (SUCCEEDED(service->QueryInterface(IFC_OmServiceCommand, (void**)&serviceCommand))) { commandState = serviceCommand->QueryState(hBrowser, &CMDGROUP_ADDRESSBAR, ADDRESSCOMMAND_VISIBLE); serviceCommand->Release(); } service->Release(); } if (E_NOTIMPL == commandState) commandState = (0 != (TBS_SHOWADDRESS & windowStyle)) ? CMDSTATE_ENABLED : CMDSTATE_DISABLED; if (0 == (ToolbarItem::stateHidden & itemStyle) && CMDSTATE_ENABLED != commandState) { Toolbar_LockUpdate(hwnd, TRUE); if (FALSE != Toolbar_ShowItem(hwnd, MAKEINTRESOURCE(index), FALSE)) { size_t nextIndex = index + 1; if (nextIndex < toolbar->items->size()) { ToolbarItem *nextItem = toolbar->items->at(nextIndex); if (NULL != nextItem && nextItem->IsEqual(TOOLITEM_FLEXSPACE, -1)) { Toolbar_ShowItem(hwnd, MAKEINTRESOURCE(nextIndex), TRUE); } } } Toolbar_LockUpdate(hwnd, FALSE); InvalidateRect(hwnd, NULL, TRUE); } } break; } } } return !fEnable; } static LRESULT Toolbar_OnEnableFancyAddress(HWND hwnd, BOOL fEnable) { UINT windowStyle = GetWindowStyle(hwnd); UINT newStyle = windowStyle; if (FALSE == fEnable) newStyle &= ~TBS_FANCYADDRESS; else newStyle |= TBS_FANCYADDRESS; if(newStyle == windowStyle) return fEnable; SetWindowLongPtr(hwnd, GWL_STYLE, newStyle); TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL != toolbar && NULL != toolbar->items) { size_t index = toolbar->items->size(); while(index--) { ToolbarItem *item = toolbar->items->at(index); if (NULL != item && item->IsEqual(TOOLITEM_ADDRESSBAR, -1)) { UINT style = (FALSE != fEnable) ? 0 : ToolbarAddress::styleAddressShowReal; item->SetStyle(hwnd, style, ToolbarAddress::styleAddressShowReal); break; } } } return !fEnable; } static LRESULT Toolbar_OnSetBrowserHost(HWND hwnd, HWND hBrowser) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return 0; toolbar->hBrowser = hBrowser; return TRUE; } static LRESULT Toolbar_OnGetImageListHeight(HWND hwnd) { TOOLBAR *toolbar = GetToolbar(hwnd); INT iconCX, iconCY; if (NULL == toolbar || NULL == toolbar->imageList || FALSE == ImageList_GetIconSize(toolbar->imageList, &iconCX, &iconCY)) { return 0; } return iconCY/TOOLBAR_ICONSTATE_COUNT; } static LRESULT Toolbar_OnGetNextTabItem(HWND hwnd, LPCSTR pszName, BOOL fPrevious) { TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return ITEM_ERR; INT index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return ITEM_ERR; if (0 == index && FALSE == fPrevious) return ITEM_ERR; UINT windowStyle = GetWindowStyle(hwnd); BOOL fTabstopOnly = (0 == (TBS_TABSTOP & windowStyle)); INT direction = (FALSE == fPrevious) ? 1 : -1; return Toolbar_GetFocusIndex(hwnd, index + direction, direction, fTabstopOnly); } static void Toolbar_OnCheckHide(HWND hwnd, BOOL fImmediate) { Toolbar_AutoHideWindow(hwnd, fImmediate); } static LRESULT Toolbar_OnGetTextLength(HWND hwnd, LPCSTR pszName, size_t *textLength) { if (NULL == textLength) return FALSE; *textLength = 0; TOOLBAR *toolbar = GetToolbar(hwnd); if (NULL == toolbar) return FALSE; size_t index = Toolbar_ResolveName(hwnd, pszName); if (ITEM_ERR == index) return FALSE; ToolbarItem *item = Toolbar_GetItem(toolbar, index); if (NULL == item) return FALSE; if (FAILED(item->GetTextLength(textLength))) return FALSE; return TRUE; } static LRESULT CALLBACK Toolbar_WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch (uMsg) { case WM_CREATE: return Toolbar_OnCreate(hwnd, (CREATESTRUCT*)lParam); case WM_DESTROY: Toolbar_OnDestroy(hwnd); break; case WM_PAINT: Toolbar_OnPaint(hwnd); return 0; case WM_PRINTCLIENT: Toolbar_OnPrintClient(hwnd, (HDC)wParam, (UINT)lParam); return 0; case WM_ERASEBKGND: return 0; case WM_WINDOWPOSCHANGED: Toolbar_OnWindowPosChanged(hwnd, (WINDOWPOS*)lParam); return 0; case WM_SETREDRAW: Toolbar_OnSetRedraw(hwnd, (BOOL)wParam); return 0; case WM_MOUSEMOVE: Toolbar_OnMouseMove(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); return 0; case WM_MOUSELEAVE: Toolbar_OnMouseLeave(hwnd); return 0; case WM_LBUTTONDOWN: Toolbar_OnLButtonDown(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); break; case WM_LBUTTONUP: Toolbar_OnLButtonUp(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); break; case WM_RBUTTONDOWN: Toolbar_OnRButtonDown(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); break; case WM_RBUTTONUP: Toolbar_OnRButtonUp(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); break; case WM_MBUTTONDOWN: Toolbar_OnMButtonDown(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); break; case WM_MBUTTONUP: Toolbar_OnMButtonUp(hwnd, (UINT)wParam, MAKEPOINTS(lParam)); break; case WM_CAPTURECHANGED: Toolbar_OnCaptureChanged(hwnd, (HWND)lParam); return 0; case WM_GETFONT: return Toolbar_OnGetFont(hwnd); case WM_SETFONT: Toolbar_OnSetFont(hwnd, (HFONT)wParam, (BOOL)LOWORD(lParam)); return 0; case WM_NOTIFY: return Toolbar_OnNotify(hwnd, (INT)wParam, (NMHDR*)lParam); case WM_COMMAND: Toolbar_OnCommand(hwnd, LOWORD(wParam), HIWORD(wParam), (HWND)lParam); return 0; case WM_ENTERMENULOOP: Toolbar_OnEnterMenuLoop(hwnd, (BOOL)wParam); return 0; case WM_EXITMENULOOP: Toolbar_OnExitMenuLoop(hwnd, (BOOL)wParam); return 0; case WM_SETFOCUS: Toolbar_OnSetFocus(hwnd, (HWND)wParam); return 0; case WM_KILLFOCUS: Toolbar_OnKillFocus(hwnd, (HWND)wParam); return 0; case WM_CONTEXTMENU: Toolbar_OnContextMenu(hwnd, (HWND)wParam, MAKEPOINTS(lParam)); return 0; case WM_KEYDOWN: Toolbar_OnKeyDown(hwnd, (INT)wParam, (UINT)lParam); return 0; case WM_KEYUP: Toolbar_OnKeyUp(hwnd, (INT)wParam, (UINT)lParam); return 0; case WM_GETDLGCODE: return Toolbar_OnGetDlgCode(hwnd, (INT)wParam, (MSG*)lParam); case WM_UPDATEUISTATE: Toolbar_OnUpdateUiState(hwnd, LOWORD(wParam), HIWORD(wParam)); return 0; case WM_ENABLE: Toolbar_OnEnable(hwnd, (BOOL)wParam); return 0; case WM_SETCURSOR: return Toolbar_OnSetCursor(hwnd, (HWND)wParam, LOWORD(lParam), HIWORD(lParam)); case WM_CTLCOLOREDIT: return Toolbar_OnColorEdit(hwnd, (HDC)wParam, (HWND)lParam); case WM_CTLCOLORSTATIC: return Toolbar_OnColorStatic(hwnd, (HDC)wParam, (HWND)lParam); case TBM_UPDATESKIN: Toolbar_OnUpdateSkin(hwnd, (BOOL)lParam); return 0; case TBM_GETIDEALHEIGHT: return Toolbar_OnGetIdealHeight(hwnd); case TBM_GETICONSIZE: return Toolbar_OnGetIconSize(hwnd, (INT)wParam, (SIZE*)lParam); case TBM_SENDCOMMAND: return Toolbar_OnSendCommand(hwnd, (INT)wParam); case TBM_DRAWICON: return Toolbar_OnDrawIcon(hwnd, (TOOLBARDRAWICONPARAM*)lParam); case TBM_GETITEMCOUNT: return Toolbar_OnGetItemCount(hwnd); case TBM_CLEAR: return Toolbar_OnClear(hwnd); case TBM_INSERTITEM: return Toolbar_OnInsertItem(hwnd, (TOOLBARINSERTITEM*)lParam); case TBM_FINDITEM: return Toolbar_OnFindItem(hwnd, (LPCSTR)lParam); case TBM_REMOVEITEM: return Toolbar_OnRemoveItem(hwnd, (LPCSTR)lParam); case TBM_SETITEMINT: return Toolbar_OnSetItemInt(hwnd, (LPCSTR)lParam, (INT)wParam); case TBM_SETITEMSTRING: return Toolbar_OnSetItemString(hwnd, (LPCSTR)lParam, (LPCWSTR)wParam); case TBM_GETBKCOLOR: return Toolbar_OnGetBkColor(hwnd); case TBM_GETFGCOLOR: return Toolbar_OnGetFgColor(hwnd); case TBM_GETTEXTCOLOR: return Toolbar_OnGetTextColor(hwnd); case TBM_GETHILITECOLOR: return Toolbar_OnGetHiliteColor(hwnd); case TBM_ENABLEITEM: return Toolbar_OnEnableItem(hwnd,(LPCSTR)lParam, (BOOL)wParam); case TBM_SHOWITEM: return Toolbar_OnShowItem(hwnd,(LPCSTR)lParam, (BOOL)wParam); case TBM_UPDATETIP: Toolbar_OnUpdateTip(hwnd); return 0; case TBM_GETTEXTMETRICS: return Toolbar_OnGetTextMetrics(hwnd, (TOOLBARTEXTMETRIC*)lParam); case TBM_GETBKBRUSH: return Toolbar_OnGetBkBrush(hwnd); case TBM_LAYOUT: return Toolbar_OnLayout(hwnd, (TOOLBARLAYOUT*)lParam); case TBM_NEXTITEM: return Toolbar_OnNextItem(hwnd, (LPCSTR)lParam, (BOOL)wParam); case TBM_GETITEMSTYLE: return Toolbar_OnGetItemStyle(hwnd, (LPCSTR)lParam, (UINT)wParam); case TBM_GETITEMCOMMAND: return Toolbar_OnGetItemCommand(hwnd, (LPCSTR)lParam); case TBM_SETITEMDESCRIPTION:return Toolbar_OnSetItemDescription(hwnd, (LPCSTR)lParam, (LPCWSTR)wParam); case TBM_GETITEMINFO: return Toolbar_OnGetItemInfo(hwnd, (LPCSTR)lParam, (TBITEMINFO*)wParam); case TBM_AUTOPOPULATE: return Toolbar_OnAutoPopulate(hwnd, (ifc_omservice*)lParam, (UINT)wParam); case TBM_ENABLEBOTTOMDOCK: return Toolbar_OnEnableBottomDock(hwnd, (BOOL)lParam); case TBM_ENABLEAUTOHIDE: return Toolbar_OnEnableAutoHide(hwnd, (BOOL)lParam); case TBM_ENABLETABSTOP: return Toolbar_OnEnableTabStop(hwnd, (BOOL)lParam); case TBM_ENABLEFORCEADDRESS: return Toolbar_OnEnableForceAddress(hwnd, (BOOL)lParam); case TBM_ENABLEFANCYADDRESS: return Toolbar_OnEnableFancyAddress(hwnd, (BOOL)lParam); case TBM_SETBROWSERHOST: return Toolbar_OnSetBrowserHost(hwnd, (HWND)lParam); case TBM_GETEDITCOLOR: return Toolbar_OnGetEditColor(hwnd); case TBM_GETEDITBKCOLOR: return Toolbar_OnGetEditBkColor(hwnd); case TBM_GETIMAGELISTHEIGHT:return Toolbar_OnGetImageListHeight(hwnd); case TBM_GETNEXTTABITEM: return Toolbar_OnGetNextTabItem(hwnd, (LPCSTR)lParam, (BOOL)wParam); case TBM_CHECKHIDE: Toolbar_OnCheckHide(hwnd, (BOOL)lParam); return 0; case TBM_GETTEXTLENGTH: return Toolbar_OnGetTextLength(hwnd, (LPCSTR)lParam, (size_t*)wParam); } return DefWindowProcW(hwnd, uMsg, wParam, lParam); } static LRESULT CALLBACK Toolbar_MouseHook(INT code, WPARAM wParam, LPARAM lParam) { TOOLBARMOUSEHOOK *hook = (TOOLBARMOUSEHOOK*)Plugin_TlsGetValue(tlsIndex); if (NULL == hook || NULL == hook->hHook) return FALSE; if (code >= 0) { MOUSEHOOKSTRUCT *mouseHook = (MOUSEHOOKSTRUCT*)lParam; if (mouseHook->hwnd != hook->hwnd) { TOOLBAR *toolbar = GetToolbar(hook->hwnd); if (NULL != toolbar) { if (0 != (TBS_HIDDEN & toolbar->flags)) { Toolbar_RemoveMouseHook(hook->hwnd, TRUE); } else if (0 == ((TBS_MENULOOP | TBS_HIDETIMERSET) & toolbar->flags)) { HWND hAncestor = GetAncestor(mouseHook->hwnd, GA_ROOT); BOOL autoHide; if (NULL == hAncestor || FALSE == IsChild(hAncestor, hook->hwnd)) { autoHide = TRUE; } else { RECT toolbarRect; if (!GetWindowRect(hook->hwnd, &toolbarRect)) SetRectEmpty(&toolbarRect); toolbarRect.top -= 6; toolbarRect.bottom += 6; autoHide = !PtInRect(&toolbarRect, mouseHook->pt); } if (FALSE != autoHide) Toolbar_AutoHideWindow(hook->hwnd, FALSE); } } } } return CallNextHookEx(hook->hHook, code, wParam, lParam); }