?? dockingcont.cpp
字號:
//this file is part of docking functionality for Notepad++
//Copyright (C)2006 Jens Lorenz <jens.plugin.npp@gmx.de>
//
//This program is free software; you can redistribute it and/or
//modify it under the terms of the GNU General Public License
//as published by the Free Software Foundation; either
//version 2 of the License, or (at your option) any later version.
//
//This program is distributed in the hope that it will be useful,
//but WITHOUT ANY WARRANTY; without even the implied warranty of
//MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
//GNU General Public License for more details.
//
//You should have received a copy of the GNU General Public License
//along with this program; if not, write to the Free Software
//Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#include "dockingResource.h"
#include "math.h"
#include "Docking.h"
#include "DockingCont.h"
#include "DropData.h"
#include "SplitterContainer.h"
#include "WindowInterface.h"
#include "SysMsg.h"
#include "ToolTip.h"
#include <Commctrl.h>
#include <shlobj.h>
#include "common_func.h"
#ifndef WH_MOUSE_LL
#define WH_MOUSE_LL 14
#endif
static HWND hWndServer = NULL;
static HHOOK hookMouse = NULL;
static LRESULT CALLBACK hookProcMouse(UINT nCode, WPARAM wParam, LPARAM lParam)
{
if(nCode < 0)
{
::CallNextHookEx(hookMouse, nCode, wParam, lParam);
return 0;
}
switch (wParam)
{
case WM_MOUSEMOVE:
case WM_NCMOUSEMOVE:
::PostMessage(hWndServer, wParam, 0, 0);
break;
case WM_LBUTTONUP:
case WM_NCLBUTTONUP:
::PostMessage(hWndServer, wParam, 0, 0);
break;
default:
break;
}
return ::CallNextHookEx(hookMouse, nCode, wParam, lParam);
}
DockingCont::DockingCont(void)
{
_isMouseOver = FALSE;
_isMouseClose = FALSE;
_isMouseDown = FALSE;
_isFloating = false;
_isTopCaption = CAPTION_TOP;
_dragFromTab = FALSE;
_hContTab = NULL;
_hDefaultTabProc = NULL;
_beginDrag = FALSE;
_prevItem = 0;
_hFont = NULL;
_bTabTTHover = FALSE;
_bCaptionTT = FALSE;
_bCapTTHover = FALSE;
_hoverMPos = posClose;
_bDrawOgLine = TRUE;
_vTbData.clear();
}
DockingCont::~DockingCont()
{
::DeleteObject(_hFont);
}
void DockingCont::doDialog(bool willBeShown, bool isFloating)
{
if (!isCreated())
{
create(IDD_CONTAINER_DLG);
_isFloating = isFloating;
if (_isFloating == true)
{
::SetWindowLong(_hSelf, GWL_STYLE, POPUP_STYLES);
::SetWindowLong(_hSelf, GWL_EXSTYLE, POPUP_EXSTYLES);
::ShowWindow(_hCaption, SW_HIDE);
}
else
{
::SetWindowLong(_hSelf, GWL_STYLE, CHILD_STYLES);
::SetWindowLong(_hSelf, GWL_EXSTYLE, CHILD_EXSTYLES);
::ShowWindow(_hCaption, SW_SHOW);
}
_hFont = ::CreateFont(14, 0, 0, 0,
FW_NORMAL, FALSE, FALSE, FALSE,
ANSI_CHARSET, OUT_DEFAULT_PRECIS,
CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY,
DEFAULT_PITCH | FF_ROMAN,
"MS Shell Dlg");
}
display(willBeShown);
}
tTbData* DockingCont::createToolbar(tTbData data, Window **ppWin)
{
tTbData *pTbData = new tTbData;
*pTbData = data;
/* force window style of client window */
::SetWindowLong(pTbData->hClient, GWL_STYLE, CHILD_STYLES);
::SetWindowLong(pTbData->hClient, GWL_EXSTYLE, CHILD_EXSTYLES);
/* restore position if plugin is in floating state */
if ((_isFloating == true) && (::SendMessage(_hContTab, TCM_GETITEMCOUNT, 0, 0) == 0))
{
reSizeToWH(pTbData->rcFloat);
}
/* set attached child window */
::SetParent(pTbData->hClient, ::GetDlgItem(_hSelf, IDC_CLIENT_TAB));
/* set names for captions and view toolbar */
viewToolbar(pTbData);
/* attach to list */
_vTbData.push_back(pTbData);
return pTbData;
}
void DockingCont::removeToolbar(tTbData TbData)
{
INT iItemCnt = 0;
/* remove from list */
for (size_t iTb = 0; iTb < _vTbData.size(); iTb++)
{
if (_vTbData[iTb]->hClient == TbData.hClient)
{
/* remove tab */
removeTab(_vTbData[iTb]);
/* free resources */
delete _vTbData[iTb];
vector<tTbData*>::iterator itr = _vTbData.begin() + iTb;
_vTbData.erase(itr);
}
}
}
tTbData* DockingCont::findToolbarByWnd(HWND hClient)
{
tTbData* pTbData = NULL;
/* find entry by handle */
for (size_t iTb = 0; iTb < _vTbData.size(); iTb++)
{
if (hClient == _vTbData[iTb]->hClient)
{
pTbData = _vTbData[iTb];
}
}
return pTbData;
}
tTbData* DockingCont::findToolbarByName(char* pszName)
{
tTbData* pTbData = NULL;
/* find entry by handle */
for (size_t iTb = 0; iTb < _vTbData.size(); iTb++)
{
if (strcmp(pszName, _vTbData[iTb]->pszName) == 0)
{
pTbData = _vTbData[iTb];
}
}
return pTbData;
}
void DockingCont::setActiveTb(tTbData* pTbData)
{
INT iItem = SearchPosInTab(pTbData);
setActiveTb(iItem);
}
void DockingCont::setActiveTb(INT iItem)
{
//if ((iItem != -1) && (iItem < ::SendMessage(_hContTab, TCM_GETITEMCOUNT, 0, 0)))
if (iItem < ::SendMessage(_hContTab, TCM_GETITEMCOUNT, 0, 0))
{
SelectTab(iItem);
}
}
INT DockingCont::getActiveTb(void)
{
return ::SendMessage(_hContTab, TCM_GETCURSEL, 0, 0);
}
tTbData* DockingCont::getDataOfActiveTb(void)
{
tTbData* pTbData = NULL;
INT iItem = getActiveTb();
if (iItem != -1)
{
TCITEM tcItem = {0};
tcItem.mask = TCIF_PARAM;
::SendMessage(_hContTab, TCM_GETITEM, iItem, (LPARAM)&tcItem);
pTbData = (tTbData*)tcItem.lParam;
}
return pTbData;
}
vector<tTbData*> DockingCont::getDataOfVisTb(void)
{
vector<tTbData*> vTbData;
TCITEM tcItem = {0};
INT iItemCnt = ::SendMessage(_hContTab, TCM_GETITEMCOUNT, 0, 0);
tcItem.mask = TCIF_PARAM;
for(INT iItem = 0; iItem < iItemCnt; iItem++)
{
::SendMessage(_hContTab, TCM_GETITEM, iItem, (LPARAM)&tcItem);
vTbData.push_back((tTbData*)tcItem.lParam);
}
return vTbData;
}
bool DockingCont::isTbVis(tTbData* data)
{
bool bRet = false;
TCITEM tcItem = {0};
INT iItemCnt = ::SendMessage(_hContTab, TCM_GETITEMCOUNT, 0, 0);
tcItem.mask = TCIF_PARAM;
for(INT iItem = 0; iItem < iItemCnt; iItem++)
{
::SendMessage(_hContTab, TCM_GETITEM, iItem, (LPARAM)&tcItem);
if (((tTbData*)tcItem.lParam) == data)
{
bRet = true;
break;
}
}
return bRet;
}
/*********************************************************************************
* Process function of caption bar
*/
LRESULT DockingCont::runProcCaption(HWND hwnd, UINT Message, WPARAM wParam, LPARAM lParam)
{
static ToolTip toolTip;
switch (Message)
{
case WM_LBUTTONDOWN:
{
_isMouseDown = TRUE;
if (isInRect(hwnd, LOWORD(lParam), HIWORD(lParam)) == posClose)
{
_isMouseClose = TRUE;
_isMouseOver = TRUE;
/* start hooking */
hWndServer = _hCaption;
if (GetVersion() & 0x80000000)
{
hookMouse = ::SetWindowsHookEx(WH_MOUSE, (HOOKPROC)hookProcMouse, _hInst, 0);
}
else
{
hookMouse = ::SetWindowsHookEx(WH_MOUSE_LL, (HOOKPROC)hookProcMouse, _hInst, 0);
}
if (!hookMouse)
{
DWORD dwError = ::GetLastError();
TCHAR str[128];
::wsprintf(str, "GetLastError() returned %lu", dwError);
::MessageBox(NULL, str, "SetWindowsHookEx(MOUSE) failed", MB_OK | MB_ICONERROR);
}
::RedrawWindow(hwnd, NULL, NULL, TRUE);
}
focusClient();
return TRUE;
}
case WM_LBUTTONUP:
{
_isMouseDown = FALSE;
if (_isMouseClose == TRUE)
{
/* end hooking */
::UnhookWindowsHookEx(hookMouse);
if (_isMouseOver == TRUE)
{
doClose();
}
_isMouseClose = FALSE;
_isMouseOver = FALSE;
}
focusClient();
return TRUE;
}
case WM_LBUTTONDBLCLK:
{
if (isInRect(hwnd, LOWORD(lParam), HIWORD(lParam)) == posCaption)
::SendMessage(_hParent, DMM_FLOATALL, 0, (LPARAM)this);
focusClient();
return TRUE;
}
case WM_MOUSEMOVE:
{
POINT pt = {0};
/* get correct cursor position */
::GetCursorPos(&pt);
::ScreenToClient(_hCaption, &pt);
if (_isMouseDown == TRUE)
{
if (_isMouseClose == FALSE)
{
/* keep sure that button is still down and within caption */
if ((wParam == MK_LBUTTON) && (isInRect(hwnd, pt.x, pt.y) == posCaption))
{
_dragFromTab = FALSE;
NotifyParent(DMM_MOVE);
_isMouseDown = FALSE;
}
else
{
_isMouseDown = FALSE;
}
}
else
{
BOOL isMouseOver = _isMouseOver;
_isMouseOver = (isInRect(hwnd, pt.x, pt.y) == posClose ? TRUE : FALSE);
/* if state is changed draw new */
if (_isMouseOver != isMouseOver)
{
::SetFocus(NULL);
::RedrawWindow(hwnd, NULL, NULL, TRUE);
}
}
}
else if (_bCapTTHover == FALSE)
{
_hoverMPos = isInRect(hwnd, LOWORD(lParam), HIWORD(lParam));
if ((_bCaptionTT == TRUE) || (_hoverMPos == posClose))
{
TRACKMOUSEEVENT tme;
tme.cbSize = sizeof(tme);
tme.hwndTrack = hwnd;
tme.dwFlags = TME_LEAVE | TME_HOVER;
tme.dwHoverTime = 1000;
_bCapTTHover = _TrackMouseEvent(&tme);
}
}
else if ((_bCapTTHover == TRUE) &&
(_hoverMPos != isInRect(hwnd, LOWORD(lParam), HIWORD(lParam))))
{
toolTip.destroy();
_bCapTTHover = FALSE;
}
return TRUE;
}
case WM_MOUSEHOVER:
{
RECT rc = {0};
POINT pt = {0};
/* get mouse position */
::GetCursorPos(&pt);
toolTip.init(_hInst, hwnd);
if (_hoverMPos == posCaption)
{
toolTip.Show(rc, _pszCaption, pt.x, pt.y + 20);
}
else
{
toolTip.Show(rc, "Close", pt.x, pt.y + 20);
}
return TRUE;
}
case WM_MOUSELEAVE:
{
toolTip.destroy();
_bCapTTHover = FALSE;
return TRUE;
}
case WM_SIZE:
{
::GetWindowRect(hwnd, &_rcCaption);
ScreenToClient(hwnd, &_rcCaption);
break;
}
case WM_SETTEXT:
{
::RedrawWindow(hwnd, NULL, NULL, TRUE);
return TRUE;
}
default:
break;
}
return ::CallWindowProc(_hDefaultCaptionProc, hwnd, Message, wParam, lParam);
}
void DockingCont::drawCaptionItem(DRAWITEMSTRUCT *pDrawItemStruct)
{
HBRUSH bgbrush = NULL;
HFONT hOldFont = NULL;
RECT rc = pDrawItemStruct->rcItem;
HDC hDc = pDrawItemStruct->hDC;
HPEN hPen = ::CreatePen(PS_SOLID, 1, ::GetSysColor(COLOR_BTNSHADOW));
BITMAP bmp = {0};
HBITMAP hBmpCur = NULL;
HBITMAP hBmpOld = NULL;
HBITMAP hBmpNew = NULL;
UINT length = strlen(_pszCaption);
INT nSavedDC = ::SaveDC(hDc);
/* begin with paint */
::SetBkMode(hDc, TRANSPARENT);
if (_isActive == TRUE) {
bgbrush = ::CreateSolidBrush(::GetSysColor(COLOR_ACTIVECAPTION));
::SetTextColor(hDc, ::GetSysColor(COLOR_CAPTIONTEXT));
} else {
bgbrush = ::CreateSolidBrush(::GetSysColor(COLOR_BTNFACE));
}
?? 快捷鍵說明
復(fù)制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -