mirror of
https://github.com/reactos/reactos.git
synced 2024-11-20 06:15:26 +00:00
f9b5ca3c79
CORE-9198 #resolve svn path=/trunk/; revision=66372
1120 lines
26 KiB
C++
1120 lines
26 KiB
C++
/*
|
|
* Copyright 2003, 2004, 2005 Martin Fuchs
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library 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
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
|
|
//
|
|
// Explorer clone
|
|
//
|
|
// window.h
|
|
//
|
|
// Martin Fuchs, 23.07.2003
|
|
//
|
|
|
|
|
|
typedef set<HWND> WindowSet;
|
|
|
|
|
|
/*
|
|
Classes are declared using "struct", not "class" because the default
|
|
access mode is "public". This way we can list the member functions in a
|
|
natural order without explicitly specifying any access mode at the begin
|
|
of the definition.
|
|
First are public constructors and destructor, then public member functions.
|
|
After that we list protected member varibables and functions. If needed,
|
|
private implemenation varibales and functions are positioned at the end.
|
|
*/
|
|
|
|
|
|
/// information structure for creation of a MDI child window
|
|
struct ChildWndInfo
|
|
{
|
|
ChildWndInfo(HWND hmdiclient)
|
|
: _hmdiclient(hmdiclient) {}
|
|
|
|
HWND _hmdiclient;
|
|
};
|
|
|
|
|
|
/**
|
|
Class Window is the base class for several C++ window wrapper classes.
|
|
Window objects are allocated from the heap. They are automatically freed
|
|
when the window gets destroyed.
|
|
*/
|
|
struct Window : public WindowHandle
|
|
{
|
|
Window(HWND hwnd);
|
|
virtual ~Window();
|
|
|
|
|
|
typedef map<HWND,Window*> WindowMap;
|
|
|
|
typedef Window* (*CREATORFUNC)(HWND);
|
|
typedef Window* (*CREATORFUNC_INFO)(HWND, const void*);
|
|
|
|
static HWND Create(CREATORFUNC creator,
|
|
DWORD dwExStyle, LPCTSTR lpClassName, LPCTSTR lpWindowName,
|
|
DWORD dwStyle, int x, int y, int w, int h,
|
|
HWND hwndParent=0, HMENU hMenu=0/*, LPVOID lpParam=0*/);
|
|
|
|
static HWND Create(CREATORFUNC_INFO creator, const void* info,
|
|
DWORD dwExStyle, LPCTSTR lpClassName, LPCTSTR lpWindowName,
|
|
DWORD dwStyle, int x, int y, int w, int h,
|
|
HWND hwndParent=0, HMENU hMenu=0/*, LPVOID lpParam=0*/);
|
|
|
|
static Window* create_mdi_child(const ChildWndInfo& info, const MDICREATESTRUCT& mcs, CREATORFUNC_INFO creator);
|
|
// static Window* create_property_sheet(struct PropertySheetDialog* ppsd, CREATORFUNC creator, const void* info);
|
|
|
|
static LRESULT CALLBACK WindowWndProc(HWND hwnd, UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
static INT_PTR CALLBACK DialogProc(HWND hwnd, UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
|
|
static Window* get_window(HWND hwnd);
|
|
#ifndef _MSC_VER
|
|
template<typename CLASS> static CLASS* get_window(HWND hwnd) {return static_cast<CLASS*>(get_window(hwnd));}
|
|
#define GET_WINDOW(CLASS, hwnd) Window::get_window<CLASS>(hwnd)
|
|
#endif
|
|
|
|
static void register_pretranslate(HWND hwnd);
|
|
static void unregister_pretranslate(HWND hwnd);
|
|
static BOOL pretranslate_msg(LPMSG pmsg);
|
|
|
|
static void register_dialog(HWND hwnd);
|
|
static void unregister_dialog(HWND hwnd);
|
|
static BOOL dispatch_dialog_msg(LPMSG pmsg);
|
|
|
|
static int MessageLoop();
|
|
|
|
|
|
LRESULT SendParent(UINT nmsg, WPARAM wparam=0, LPARAM lparam=0);
|
|
LRESULT PostParent(UINT nmsg, WPARAM wparam=0, LPARAM lparam=0);
|
|
|
|
static void CancelModes();
|
|
|
|
|
|
protected:
|
|
virtual LRESULT Init(LPCREATESTRUCT pcs); // WM_CREATE processing
|
|
virtual LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
virtual int Command(int id, int code); // WM_COMMAND processing
|
|
virtual int Notify(int id, NMHDR* pnmh); // WM_NOTIFY processing
|
|
|
|
static Window* create_controller(HWND hwnd);
|
|
|
|
|
|
static WindowMap s_wnd_map;
|
|
|
|
static const void* s_new_info;
|
|
static CREATORFUNC s_window_creator;
|
|
|
|
|
|
/// structure for managing critical sections as static class information in struct Window
|
|
struct StaticWindowData {
|
|
CritSect _map_crit_sect;
|
|
CritSect _create_crit_sect;
|
|
};
|
|
|
|
static StaticWindowData& GetStaticWindowData();
|
|
|
|
|
|
// MDI child creation
|
|
static HHOOK s_hcbtHook;
|
|
static LRESULT CALLBACK MDICBTHookProc(int code, WPARAM wparam, LPARAM lparam);
|
|
static LRESULT CALLBACK PropSheetCBTHookProc(int code, WPARAM wparam, LPARAM lparam);
|
|
|
|
static WindowSet s_pretranslate_windows;
|
|
static WindowSet s_dialogs;
|
|
};
|
|
|
|
#ifdef UNICODE
|
|
#define NFR_CURRENT NFR_UNICODE
|
|
#else
|
|
#define NFR_CURRENT NFR_ANSI
|
|
#endif
|
|
|
|
|
|
#ifdef _MSC_VER
|
|
template<typename CLASS> struct GetWindowHelper
|
|
{
|
|
static CLASS* get_window(HWND hwnd) {
|
|
return static_cast<CLASS*>(Window::get_window(hwnd));
|
|
}
|
|
};
|
|
#define GET_WINDOW(CLASS, hwnd) GetWindowHelper<CLASS>::get_window(hwnd)
|
|
#endif
|
|
|
|
|
|
/// dynamic casting of Window pointers
|
|
template<typename CLASS> struct TypeCheck
|
|
{
|
|
static CLASS* dyn_cast(Window* wnd)
|
|
{return dynamic_cast<CLASS*>(wnd);}
|
|
};
|
|
|
|
#define WINDOW_DYNAMIC_CAST(CLASS, hwnd) \
|
|
TypeCheck<CLASS>::dyn_cast(Window::get_window(hwnd))
|
|
|
|
|
|
/**
|
|
SubclassedWindow is used to wrap already existing window handles
|
|
into C++ Window objects. To construct a object, use the "new" operator
|
|
to put it in the heap. It is automatically freed, when the window
|
|
gets destroyed.
|
|
*/
|
|
struct SubclassedWindow : public Window
|
|
{
|
|
typedef Window super;
|
|
|
|
SubclassedWindow(HWND);
|
|
|
|
protected:
|
|
WNDPROC _orgWndProc;
|
|
|
|
static LRESULT CALLBACK SubclassedWndProc(HWND hwnd, UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
|
|
virtual LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
virtual int Command(int id, int code);
|
|
virtual int Notify(int id, NMHDR* pnmh);
|
|
};
|
|
|
|
|
|
/// template class used in macro WINDOW_CREATOR to define the creater functions for Window objects
|
|
template<typename WND_CLASS> struct WindowCreator
|
|
{
|
|
static WND_CLASS* window_creator(HWND hwnd)
|
|
{
|
|
return new WND_CLASS(hwnd);
|
|
}
|
|
};
|
|
|
|
#define WINDOW_CREATOR(WND_CLASS) \
|
|
((Window::CREATORFUNC) WindowCreator<WND_CLASS>::window_creator)
|
|
|
|
|
|
/// template class used in macro WINDOW_CREATOR_INFO to the define creater functions for Window objects with additional creation information
|
|
template<typename WND_CLASS, typename INFO_CLASS> struct WindowCreatorInfo
|
|
{
|
|
static WND_CLASS* window_creator(HWND hwnd, const void* info)
|
|
{
|
|
return new WND_CLASS(hwnd, *static_cast<const INFO_CLASS*>(info));
|
|
}
|
|
};
|
|
|
|
#define WINDOW_CREATOR_INFO(WND_CLASS, INFO_CLASS) \
|
|
((Window::CREATORFUNC_INFO) WindowCreatorInfo<WND_CLASS, INFO_CLASS>::window_creator)
|
|
|
|
|
|
/**
|
|
WindowClass is a neat wrapper for RegisterClassEx().
|
|
Just construct a WindowClass object, override the attributes you want
|
|
to change, then call Register() or simply request the ATOM value to
|
|
register the window class. You don't have to worry calling Register()
|
|
more than once. It checks if, the class has already been registered.
|
|
*/
|
|
struct WindowClass : public WNDCLASSEX
|
|
{
|
|
WindowClass(LPCTSTR classname, UINT style=0, WNDPROC wndproc=Window::WindowWndProc);
|
|
|
|
ATOM Register()
|
|
{
|
|
if (!_atomClass)
|
|
_atomClass = RegisterClassEx(this);
|
|
|
|
return _atomClass;
|
|
}
|
|
|
|
operator ATOM() {return Register();}
|
|
|
|
// return LPCTSTR for the CreateWindowEx() parameter
|
|
operator LPCTSTR() {return (LPCTSTR)(int)Register();}
|
|
|
|
protected:
|
|
ATOM _atomClass;
|
|
};
|
|
|
|
/// window class with gray background color
|
|
struct BtnWindowClass : public WindowClass
|
|
{
|
|
BtnWindowClass(LPCTSTR classname, UINT style=0, WNDPROC wndproc=Window::WindowWndProc)
|
|
: WindowClass(classname, style, wndproc)
|
|
{
|
|
hbrBackground = (HBRUSH)(COLOR_BTNFACE+1);
|
|
}
|
|
};
|
|
|
|
/// window class with specified icon from resources
|
|
struct IconWindowClass : public WindowClass
|
|
{
|
|
IconWindowClass(LPCTSTR classname, UINT nid, UINT style=0, WNDPROC wndproc=Window::WindowWndProc);
|
|
};
|
|
|
|
|
|
// private message constants
|
|
#define PM_DISPATCH_COMMAND (WM_APP+0x00)
|
|
#define PM_TRANSLATE_MSG (WM_APP+0x01)
|
|
|
|
|
|
#define SPLIT_WIDTH 5
|
|
#define DEFAULT_SPLIT_POS 300
|
|
#define COLOR_SPLITBAR LTGRAY_BRUSH
|
|
|
|
|
|
/// menu info structure
|
|
struct MenuInfo
|
|
{
|
|
HMENU _hMenuView;
|
|
};
|
|
|
|
#define PM_FRM_GET_MENUINFO (WM_APP+0x02)
|
|
|
|
#define Frame_GetMenuInfo(hwnd) ((MenuInfo*)SNDMSG(hwnd, PM_FRM_GET_MENUINFO, 0, 0))
|
|
|
|
|
|
/**
|
|
PreTranslateWindow is used to register windows to be called by Window::pretranslate_msg().
|
|
This way you get PM_TRANSLATE_MSG messages before the message loop dispatches messages.
|
|
You can then for example use TranslateAccelerator() to implement key shortcuts.
|
|
*/
|
|
struct PreTranslateWindow : public Window
|
|
{
|
|
typedef Window super;
|
|
|
|
PreTranslateWindow(HWND);
|
|
~PreTranslateWindow();
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
Class ChildWindow represents MDI child windows.
|
|
It is used with class MainFrame.
|
|
*/
|
|
struct ChildWindow : public PreTranslateWindow
|
|
{
|
|
typedef PreTranslateWindow super;
|
|
|
|
ChildWindow(HWND hwnd, const ChildWndInfo& info);
|
|
|
|
static ChildWindow* create(const ChildWndInfo& info, const RECT& rect, CREATORFUNC_INFO creator,
|
|
LPCTSTR classname, LPCTSTR title=NULL, DWORD style=0);
|
|
|
|
bool go_to(LPCTSTR url);
|
|
|
|
protected:
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
|
|
virtual void resize_children(int cx, int cy);
|
|
virtual String jump_to_int(LPCTSTR url) = 0;
|
|
|
|
protected:
|
|
MenuInfo*_menu_info;
|
|
|
|
WindowHandle _left_hwnd;
|
|
WindowHandle _right_hwnd;
|
|
int _focus_pane; // 0: left 1: right
|
|
|
|
int _split_pos;
|
|
int _last_split;
|
|
|
|
HWND _hwndFrame;
|
|
String _statusText;
|
|
String _url;
|
|
|
|
stack<String> _url_history;
|
|
|
|
void set_url(LPCTSTR url);
|
|
};
|
|
|
|
#define PM_SETSTATUSTEXT (WM_APP+0x1E)
|
|
|
|
|
|
/**
|
|
The class DialogWindow implements modeless dialogs, which are managed by
|
|
Window::dispatch_dialog_msg() in Window::MessageLoop().
|
|
A DialogWindow object should be constructed by calling Window::Create()
|
|
and specifying the class using the WINDOW_CREATOR() macro.
|
|
*/
|
|
struct DialogWindow : public Window
|
|
{
|
|
typedef Window super;
|
|
|
|
DialogWindow(HWND hwnd)
|
|
: super(hwnd)
|
|
{
|
|
register_dialog(hwnd);
|
|
}
|
|
|
|
~DialogWindow()
|
|
{
|
|
unregister_dialog(_hwnd);
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
The class Dialog implements modal dialogs.
|
|
A Dialog object should be constructed by calling Dialog::DoModal()
|
|
and specifying the class using the WINDOW_CREATOR() macro.
|
|
*/
|
|
struct Dialog : public Window
|
|
{
|
|
typedef Window super;
|
|
|
|
Dialog(HWND);
|
|
~Dialog();
|
|
|
|
static int DoModal(UINT nid, CREATORFUNC creator, HWND hwndParent=0);
|
|
static int DoModal(UINT nid, CREATORFUNC_INFO creator, const void* info, HWND hwndParent=0);
|
|
|
|
protected:
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
int Command(int id, int code);
|
|
};
|
|
|
|
|
|
#define PM_FRM_CALC_CLIENT (WM_APP+0x03)
|
|
#define Frame_CalcFrameClient(hwnd, prt) ((BOOL)SNDMSG(hwnd, PM_FRM_CALC_CLIENT, 0, (LPARAM)(PRECT)prt))
|
|
|
|
#define PM_JUMP_TO_URL (WM_APP+0x25)
|
|
#define PM_URL_CHANGED (WM_APP+0x26)
|
|
|
|
|
|
struct PropSheetPage : public PROPSHEETPAGE
|
|
{
|
|
PropSheetPage(UINT nid, Window::CREATORFUNC dlg_creator);
|
|
|
|
void init(struct PropertySheetDialog*);
|
|
|
|
protected:
|
|
friend struct PropSheetPageDlg;
|
|
|
|
Window::CREATORFUNC _dlg_creator;
|
|
};
|
|
|
|
|
|
/// Property Sheet dialog
|
|
struct PropertySheetDialog : public PROPSHEETHEADER
|
|
{
|
|
PropertySheetDialog(HWND owner);
|
|
|
|
void add(PropSheetPage& psp);
|
|
int DoModal(int start_page=0);
|
|
|
|
HWND GetCurrentPage();
|
|
|
|
protected:
|
|
typedef vector<PROPSHEETPAGE> Vector;
|
|
Vector _pages;
|
|
HWND _hwnd;
|
|
};
|
|
|
|
|
|
/// Property Sheet Page (inner dialog)
|
|
struct PropSheetPageDlg : public Dialog
|
|
{
|
|
typedef Dialog super;
|
|
|
|
PropSheetPageDlg(HWND);
|
|
|
|
protected:
|
|
friend struct PropertySheetDialog;
|
|
friend struct PropSheetPage;
|
|
|
|
static INT_PTR CALLBACK DialogProc(HWND hwnd, UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
|
|
int Command(int id, int code);
|
|
};
|
|
|
|
|
|
/*
|
|
/// Property Sheet Dialog (outer dialog)
|
|
struct PropertySheetDlg : public SubclassedWindow
|
|
{
|
|
typedef SubclassedWindow super;
|
|
|
|
PropertySheetDlg(HWND hwnd) : super(hwnd) {}
|
|
};
|
|
*/
|
|
|
|
|
|
// Layouting of resizable windows
|
|
|
|
/// Flags to specify how to move and resize controls when resizing their parent window
|
|
enum RESIZE_FLAGS {
|
|
MOVE_LEFT = 0x1,
|
|
MOVE_RIGHT = 0x2,
|
|
MOVE_TOP = 0x4,
|
|
MOVE_BOTTOM = 0x8,
|
|
|
|
MOVE_X = MOVE_LEFT | MOVE_RIGHT,
|
|
MOVE_Y = MOVE_TOP | MOVE_BOTTOM,
|
|
RESIZE_X= MOVE_RIGHT,
|
|
RESIZE_Y= MOVE_BOTTOM,
|
|
|
|
MOVE = MOVE_X | MOVE_Y,
|
|
RESIZE = RESIZE_X | RESIZE_Y
|
|
};
|
|
|
|
/// structure to assign RESIZE_FLAGS to dialogs control
|
|
struct ResizeEntry
|
|
{
|
|
ResizeEntry(UINT id, int flags)
|
|
: _id(id), _flags(flags) {}
|
|
|
|
ResizeEntry(HWND hwnd, int flags)
|
|
: _id(GetDlgCtrlID(hwnd)), _flags(flags) {}
|
|
|
|
UINT _id;
|
|
int _flags;
|
|
};
|
|
|
|
|
|
/// Management of controls in resizable dialogs
|
|
struct ResizeManager : public std::list<ResizeEntry>
|
|
{
|
|
typedef std::list<ResizeEntry> super;
|
|
|
|
ResizeManager(HWND hwnd);
|
|
|
|
void Add(UINT id, int flags)
|
|
{push_back(ResizeEntry(id, flags));}
|
|
|
|
void Add(HWND hwnd, int flags)
|
|
{push_back(ResizeEntry(hwnd, flags));}
|
|
|
|
void HandleSize(int cx, int cy);
|
|
void Resize(int dx, int dy);
|
|
|
|
void SetMinMaxInfo(LPMINMAXINFO lpmmi)
|
|
{
|
|
lpmmi->ptMinTrackSize.x = _min_wnd_size.cx;
|
|
lpmmi->ptMinTrackSize.y = _min_wnd_size.cy;
|
|
}
|
|
|
|
SIZE _min_wnd_size;
|
|
|
|
protected:
|
|
HWND _hwnd;
|
|
SIZE _last_size;
|
|
};
|
|
|
|
|
|
/// Controller base template class for resizable dialogs
|
|
template<typename BASE> struct ResizeController : public BASE
|
|
{
|
|
typedef BASE super;
|
|
|
|
ResizeController(HWND hwnd)
|
|
: super(hwnd),
|
|
_resize_mgr(hwnd)
|
|
{
|
|
}
|
|
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
switch(nmsg) {
|
|
case PM_FRM_CALC_CLIENT:
|
|
GetClientSpace((PRECT)lparam);
|
|
return TRUE;
|
|
|
|
case WM_SIZE:
|
|
if (wparam != SIZE_MINIMIZED)
|
|
_resize_mgr.HandleSize(LOWORD(lparam), HIWORD(lparam));
|
|
goto def;
|
|
|
|
case WM_GETMINMAXINFO:
|
|
_resize_mgr.SetMinMaxInfo((LPMINMAXINFO)lparam);
|
|
goto def;
|
|
|
|
default: def:
|
|
return super::WndProc(nmsg, wparam, lparam);
|
|
}
|
|
}
|
|
|
|
virtual void GetClientSpace(PRECT prect)
|
|
{
|
|
if (!IsIconic(this->_hwnd)) {
|
|
GetClientRect(this->_hwnd, prect);
|
|
} else {
|
|
WINDOWPLACEMENT wp;
|
|
GetWindowPlacement(this->_hwnd, &wp);
|
|
prect->left = prect->top = 0;
|
|
prect->right = wp.rcNormalPosition.right-wp.rcNormalPosition.left-
|
|
2*(GetSystemMetrics(SM_CXSIZEFRAME)+GetSystemMetrics(SM_CXEDGE));
|
|
prect->bottom = wp.rcNormalPosition.bottom-wp.rcNormalPosition.top-
|
|
2*(GetSystemMetrics(SM_CYSIZEFRAME)+GetSystemMetrics(SM_CYEDGE))-
|
|
GetSystemMetrics(SM_CYCAPTION)-GetSystemMetrics(SM_CYMENUSIZE);
|
|
}
|
|
}
|
|
|
|
protected:
|
|
ResizeManager _resize_mgr;
|
|
};
|
|
|
|
|
|
/**
|
|
This class constructs button controls.
|
|
The button will remain existent when the C++ Button object is destroyed.
|
|
There is no conjunction between C++ object and windows control life time.
|
|
*/
|
|
struct Button : public WindowHandle
|
|
{
|
|
Button(HWND parent, LPCTSTR text, int left, int top, int width, int height,
|
|
int id, DWORD flags=WS_VISIBLE|WS_CHILD|BS_PUSHBUTTON, DWORD exStyle=0);
|
|
};
|
|
|
|
|
|
/**
|
|
This class constructs static controls.
|
|
The control will remain existent when the C++ object is destroyed.
|
|
There is no conjunction between C++ object and windows control life time.
|
|
*/
|
|
struct Static : public WindowHandle
|
|
{
|
|
Static(HWND parent, LPCTSTR text, int left, int top, int width, int height,
|
|
int id, DWORD flags=WS_VISIBLE|WS_CHILD|SS_SIMPLE, DWORD ex_flags=0);
|
|
};
|
|
|
|
|
|
// control color message routing for ColorStatic and HyperlinkCtrl
|
|
|
|
#define PM_DISPATCH_CTLCOLOR (WM_APP+0x08)
|
|
|
|
template<typename BASE> struct CtlColorParent : public BASE
|
|
{
|
|
typedef BASE super;
|
|
|
|
CtlColorParent(HWND hwnd)
|
|
: super(hwnd) {}
|
|
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
switch(nmsg) {
|
|
case WM_CTLCOLOR:
|
|
case WM_CTLCOLORBTN:
|
|
case WM_CTLCOLORDLG:
|
|
case WM_CTLCOLORSCROLLBAR:
|
|
case WM_CTLCOLORSTATIC: {
|
|
HWND hctl = (HWND) lparam;
|
|
return SendMessage(hctl, PM_DISPATCH_CTLCOLOR, wparam, nmsg);
|
|
}
|
|
|
|
default:
|
|
return super::WndProc(nmsg, wparam, lparam);
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
#define PM_DISPATCH_DRAWITEM (WM_APP+0x09)
|
|
|
|
/// draw message routing for ColorButton and PictureButton
|
|
template<typename BASE> struct OwnerDrawParent : public BASE
|
|
{
|
|
typedef BASE super;
|
|
|
|
OwnerDrawParent(HWND hwnd)
|
|
: super(hwnd) {}
|
|
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
switch(nmsg) {
|
|
case WM_DRAWITEM:
|
|
if (wparam) { // should there be drawn a control?
|
|
HWND hctl = GetDlgItem(this->_hwnd, wparam);
|
|
|
|
if (hctl)
|
|
return SendMessage(hctl, PM_DISPATCH_DRAWITEM, wparam, lparam);
|
|
} /*else // or is it a menu entry?
|
|
; */
|
|
|
|
return 0;
|
|
|
|
default:
|
|
return super::WndProc(nmsg, wparam, lparam);
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
/**
|
|
Subclass button controls to draw them by using PM_DISPATCH_DRAWITEM
|
|
The owning window should use the OwnerDrawParent template to route owner draw messages to the buttons.
|
|
*/
|
|
struct OwnerdrawnButton : public SubclassedWindow
|
|
{
|
|
typedef SubclassedWindow super;
|
|
|
|
OwnerdrawnButton(HWND hwnd)
|
|
: super(hwnd) {}
|
|
|
|
protected:
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
|
|
virtual void DrawItem(LPDRAWITEMSTRUCT dis) = 0;
|
|
};
|
|
|
|
extern void DrawGrayText(HDC hdc, LPRECT pRect, LPCTSTR text, int dt_flags);
|
|
|
|
|
|
/**
|
|
Subclass button controls to paint colored text labels.
|
|
The owning window should use the OwnerDrawParent template to route owner draw messages to the buttons.
|
|
*/
|
|
/* not yet used
|
|
struct ColorButton : public OwnerdrawnButton
|
|
{
|
|
typedef OwnerdrawnButton super;
|
|
|
|
ColorButton(HWND hwnd, COLORREF textColor)
|
|
: super(hwnd), _textColor(textColor) {}
|
|
|
|
protected:
|
|
virtual void DrawItem(LPDRAWITEMSTRUCT dis);
|
|
|
|
COLORREF _textColor;
|
|
};
|
|
*/
|
|
|
|
|
|
struct FlatButton : public OwnerdrawnButton
|
|
{
|
|
typedef OwnerdrawnButton super;
|
|
|
|
FlatButton(HWND hwnd)
|
|
: super(hwnd), _active(false) {}
|
|
|
|
FlatButton(HWND owner, int id)
|
|
: super(GetDlgItem(owner, IDOK)), _active(false) {}
|
|
|
|
protected:
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
virtual void DrawItem(LPDRAWITEMSTRUCT dis);
|
|
|
|
COLORREF _textColor;
|
|
COLORREF _activeColor;
|
|
bool _active;
|
|
};
|
|
|
|
|
|
/**
|
|
Subclass button controls to paint pictures left to the labels.
|
|
The buttons should have set the style bit BS_OWNERDRAW.
|
|
The owning window should use the OwnerDrawParent template to route owner draw messages to the buttons.
|
|
*/
|
|
struct PictureButton : public OwnerdrawnButton
|
|
{
|
|
typedef OwnerdrawnButton super;
|
|
|
|
PictureButton(HWND hwnd, HICON hIcon, HBRUSH hbrush=GetSysColorBrush(COLOR_BTNFACE), bool flat=false)
|
|
: super(hwnd), _hIcon(hIcon), _hBmp(0), _hBrush(hbrush), _flat(flat)
|
|
{
|
|
_cx = 16;
|
|
_cy = 16;
|
|
}
|
|
|
|
PictureButton(HWND hparent, int id, HICON hIcon, HBRUSH hbrush=GetSysColorBrush(COLOR_BTNFACE), bool flat=false)
|
|
: super(GetDlgItem(hparent, id)), _hIcon(hIcon), _hBmp(0), _hBrush(hbrush), _flat(flat)
|
|
{
|
|
_cx = 16;
|
|
_cy = 16;
|
|
}
|
|
|
|
PictureButton(HWND hwnd, HBITMAP hBmp, HBRUSH hbrush=GetSysColorBrush(COLOR_BTNFACE), bool flat=false)
|
|
: super(hwnd), _hIcon(0), _hBmp(hBmp), _hBrush(hbrush), _flat(flat)
|
|
{
|
|
BITMAP bmp;
|
|
GetObject(hBmp, sizeof(bmp), &bmp);
|
|
_cx = bmp.bmWidth;
|
|
_cy = bmp.bmHeight;
|
|
}
|
|
|
|
PictureButton(HWND hparent, int id, HBITMAP hBmp, HBRUSH hbrush=GetSysColorBrush(COLOR_BTNFACE), bool flat=false)
|
|
: super(GetDlgItem(hparent, id)), _hIcon(0), _hBmp(hBmp), _hBrush(hbrush), _flat(flat)
|
|
{
|
|
BITMAP bmp;
|
|
GetObject(hBmp, sizeof(bmp), &bmp);
|
|
_cx = bmp.bmWidth;
|
|
_cy = bmp.bmHeight;
|
|
}
|
|
|
|
protected:
|
|
virtual void DrawItem(LPDRAWITEMSTRUCT dis);
|
|
|
|
HICON _hIcon;
|
|
HBITMAP _hBmp;
|
|
HBRUSH _hBrush;
|
|
|
|
int _cx;
|
|
int _cy;
|
|
|
|
bool _flat;
|
|
};
|
|
|
|
|
|
struct ColorStatic : public SubclassedWindow
|
|
{
|
|
typedef SubclassedWindow super;
|
|
|
|
ColorStatic(HWND hwnd, COLORREF textColor=RGB(255,0,0), HBRUSH hbrush_bkgnd=0, HFONT hfont=0)
|
|
: super(hwnd),
|
|
_textColor(textColor),
|
|
_hbrush_bkgnd(hbrush_bkgnd),
|
|
_hfont(hfont)
|
|
{
|
|
}
|
|
|
|
ColorStatic(HWND owner, int id, COLORREF textColor=RGB(255,0,0), HBRUSH hbrush_bkgnd=0, HFONT hfont=0)
|
|
: super(GetDlgItem(owner, id)),
|
|
_textColor(textColor),
|
|
_hbrush_bkgnd(hbrush_bkgnd),
|
|
_hfont(hfont)
|
|
{
|
|
}
|
|
|
|
protected:
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
if (nmsg == PM_DISPATCH_CTLCOLOR) {
|
|
HDC hdc = (HDC) wparam;
|
|
|
|
SetTextColor(hdc, _textColor);
|
|
|
|
if (_hfont)
|
|
SelectFont(hdc, _hfont);
|
|
|
|
if (_hbrush_bkgnd)
|
|
return (LRESULT)_hbrush_bkgnd;
|
|
else {
|
|
SetBkMode(hdc, TRANSPARENT);
|
|
return (LRESULT)GetStockBrush(HOLLOW_BRUSH);
|
|
}
|
|
} else
|
|
return super::WndProc(nmsg, wparam, lparam);
|
|
}
|
|
|
|
COLORREF _textColor;
|
|
HBRUSH _hbrush_bkgnd;
|
|
HFONT _hfont;
|
|
};
|
|
|
|
|
|
/// Hyperlink Controls
|
|
|
|
struct HyperlinkCtrl : public SubclassedWindow
|
|
{
|
|
typedef SubclassedWindow super;
|
|
|
|
HyperlinkCtrl(HWND hwnd, COLORREF colorLink=RGB(0,0,255), COLORREF colorVisited=RGB(128,0,128));
|
|
HyperlinkCtrl(HWND owner, int id, COLORREF colorLink=RGB(0,0,255), COLORREF colorVisited=RGB(128,0,128));
|
|
|
|
~HyperlinkCtrl();
|
|
|
|
String _cmd;
|
|
|
|
protected:
|
|
COLORREF _textColor;
|
|
COLORREF _colorVisited;
|
|
HFONT _hfont;
|
|
HCURSOR _crsr_link;
|
|
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam);
|
|
|
|
void init();
|
|
|
|
bool LaunchLink()
|
|
{
|
|
if (!_cmd.empty()) {
|
|
HINSTANCE hinst = ShellExecute(GetParent(_hwnd), _T("open"), _cmd, 0, 0, SW_SHOWNORMAL);
|
|
return (int)hinst > HINSTANCE_ERROR;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
};
|
|
|
|
|
|
/// encapsulation of tool tip controls
|
|
struct ToolTip : public WindowHandle
|
|
{
|
|
typedef WindowHandle super;
|
|
|
|
ToolTip(HWND owner);
|
|
|
|
void activate(BOOL active=TRUE)
|
|
{
|
|
SendMessage(_hwnd, TTM_ACTIVATE, active, 0);
|
|
}
|
|
|
|
void add(HWND hparent, HWND htool, LPCTSTR txt=LPSTR_TEXTCALLBACK, LPARAM lparam=0)
|
|
{
|
|
TOOLINFO ti = {
|
|
sizeof(TOOLINFO), TTF_SUBCLASS|TTF_IDISHWND|TTF_TRANSPARENT, hparent, (UINT)htool,
|
|
{0,0,0,0}, 0, (LPTSTR)txt, lparam
|
|
};
|
|
|
|
#ifdef UNICODE ///@todo Why is it neccesary to try both TTM_ADDTOOLW and TTM_ADDTOOLW ?!
|
|
if (!SendMessage(_hwnd, TTM_ADDTOOLW, 0, (LPARAM)&ti))
|
|
SendMessage(_hwnd, TTM_ADDTOOLA, 0, (LPARAM)&ti);
|
|
#else
|
|
if (!SendMessage(_hwnd, TTM_ADDTOOLA, 0, (LPARAM)&ti))
|
|
SendMessage(_hwnd, TTM_ADDTOOLW, 0, (LPARAM)&ti);
|
|
#endif
|
|
}
|
|
|
|
void add(HWND hparent, UINT id, const RECT& rect, LPCTSTR txt=LPSTR_TEXTCALLBACK, LPARAM lparam=0)
|
|
{
|
|
TOOLINFO ti = {
|
|
sizeof(TOOLINFO), TTF_SUBCLASS|TTF_TRANSPARENT, hparent, id,
|
|
{rect.left,rect.top,rect.right,rect.bottom}, 0, (LPTSTR)txt, lparam
|
|
};
|
|
|
|
#ifdef UNICODE
|
|
if (!SendMessage(_hwnd, TTM_ADDTOOLW, 0, (LPARAM)&ti))
|
|
SendMessage(_hwnd, TTM_ADDTOOLA, 0, (LPARAM)&ti);
|
|
#else
|
|
if (!SendMessage(_hwnd, TTM_ADDTOOLA, 0, (LPARAM)&ti))
|
|
SendMessage(_hwnd, TTM_ADDTOOLW, 0, (LPARAM)&ti);
|
|
#endif
|
|
}
|
|
|
|
void remove(HWND hparent, HWND htool)
|
|
{
|
|
TOOLINFO ti = {
|
|
sizeof(TOOLINFO), TTF_IDISHWND, hparent, (UINT)htool,
|
|
{0,0,0,0}, 0, 0, 0
|
|
};
|
|
|
|
SendMessage(_hwnd, TTM_DELTOOL, 0, (LPARAM)&ti);
|
|
}
|
|
|
|
void remove(HWND hparent, UINT id)
|
|
{
|
|
TOOLINFO ti = {
|
|
sizeof(TOOLINFO), 0, hparent, id,
|
|
{0,0,0,0}, 0, 0, 0
|
|
};
|
|
|
|
SendMessage(_hwnd, TTM_DELTOOL, 0, (LPARAM)&ti);
|
|
}
|
|
};
|
|
|
|
|
|
inline int ListView_GetItemData(HWND list_ctrl, int idx)
|
|
{
|
|
LV_ITEM item;
|
|
|
|
item.mask = LVIF_PARAM;
|
|
item.iItem = idx;
|
|
|
|
if (!ListView_GetItem(list_ctrl, &item))
|
|
return 0;
|
|
|
|
return item.lParam;
|
|
}
|
|
|
|
inline int ListView_FindItemPara(HWND list_ctrl, LPARAM param)
|
|
{
|
|
LVFINDINFO fi;
|
|
|
|
fi.flags = LVFI_PARAM;
|
|
fi.lParam = param;
|
|
|
|
return ListView_FindItem(list_ctrl, (unsigned)-1, &fi);
|
|
}
|
|
|
|
inline int ListView_GetFocusedItem(HWND list_ctrl)
|
|
{
|
|
int idx = ListView_GetItemCount(list_ctrl);
|
|
|
|
while(--idx >= 0)
|
|
if (ListView_GetItemState(list_ctrl, idx, LVIS_FOCUSED))
|
|
break;
|
|
|
|
return idx;
|
|
}
|
|
|
|
|
|
/// sorting of list controls
|
|
struct ListSort : public WindowHandle
|
|
{
|
|
ListSort(HWND hwndListview, PFNLVCOMPARE compare_fct);
|
|
|
|
void toggle_sort(int idx);
|
|
void sort();
|
|
|
|
int _sort_crit;
|
|
bool _direction;
|
|
|
|
protected:
|
|
PFNLVCOMPARE _compare_fct;
|
|
|
|
static int CALLBACK CompareFunc(LPARAM lparam1, LPARAM lparam2, LPARAM lparamSort);
|
|
};
|
|
|
|
|
|
inline LPARAM TreeView_GetItemData(HWND hwndTreeView, HTREEITEM hItem)
|
|
{
|
|
TVITEM tvItem;
|
|
|
|
tvItem.mask = TVIF_PARAM;
|
|
tvItem.hItem = hItem;
|
|
|
|
if (!TreeView_GetItem(hwndTreeView, &tvItem))
|
|
return 0;
|
|
|
|
return tvItem.lParam;
|
|
}
|
|
|
|
|
|
enum {TRAYBUTTON_LEFT=0, TRAYBUTTON_RIGHT, TRAYBUTTON_MIDDLE};
|
|
|
|
#define PM_TRAYICON (WM_APP+0x20)
|
|
|
|
#define WINMSG_TASKBARCREATED TEXT("TaskbarCreated")
|
|
|
|
#define WINMSG_SHELLHOOK TEXT("SHELLHOOK")
|
|
|
|
|
|
struct TrayIcon
|
|
{
|
|
TrayIcon(HWND hparent, UINT id)
|
|
: _hparent(hparent), _id(id) {}
|
|
|
|
~TrayIcon()
|
|
{Remove();}
|
|
|
|
void Add(HICON hIcon, LPCTSTR tooltip=NULL)
|
|
{Set(NIM_ADD, _id, hIcon, tooltip);}
|
|
|
|
void Modify(HICON hIcon, LPCTSTR tooltip=NULL)
|
|
{Set(NIM_MODIFY, _id, hIcon, tooltip);}
|
|
|
|
void Remove()
|
|
{
|
|
NOTIFYICONDATA nid = {
|
|
sizeof(NOTIFYICONDATA), // cbSize
|
|
_hparent, // hWnd
|
|
_id, // uID
|
|
};
|
|
|
|
Shell_NotifyIcon(NIM_DELETE, &nid);
|
|
}
|
|
|
|
protected:
|
|
HWND _hparent;
|
|
UINT _id;
|
|
|
|
void Set(DWORD dwMessage, UINT id, HICON hIcon, LPCTSTR tooltip=NULL)
|
|
{
|
|
NOTIFYICONDATA nid = {
|
|
sizeof(NOTIFYICONDATA), // cbSize
|
|
_hparent, // hWnd
|
|
id, // uID
|
|
NIF_MESSAGE|NIF_ICON, // uFlags
|
|
PM_TRAYICON, // uCallbackMessage
|
|
hIcon // hIcon
|
|
};
|
|
|
|
if (tooltip)
|
|
lstrcpyn(nid.szTip, tooltip, COUNTOF(nid.szTip));
|
|
|
|
if (nid.szTip[0])
|
|
nid.uFlags |= NIF_TIP;
|
|
|
|
Shell_NotifyIcon(dwMessage, &nid);
|
|
}
|
|
};
|
|
|
|
|
|
template<typename BASE> struct TrayIconControllerTemplate : public BASE
|
|
{
|
|
typedef BASE super;
|
|
|
|
TrayIconControllerTemplate(HWND hwnd) : BASE(hwnd),
|
|
WM_TASKBARCREATED(RegisterWindowMessage(WINMSG_TASKBARCREATED))
|
|
{
|
|
}
|
|
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
if (nmsg == PM_TRAYICON) {
|
|
switch(lparam) {
|
|
case WM_MOUSEMOVE:
|
|
TrayMouseOver(wparam);
|
|
break;
|
|
|
|
case WM_LBUTTONDOWN:
|
|
TrayClick(wparam, TRAYBUTTON_LEFT);
|
|
break;
|
|
|
|
case WM_LBUTTONDBLCLK:
|
|
TrayDblClick(wparam, TRAYBUTTON_LEFT);
|
|
break;
|
|
|
|
case WM_RBUTTONDOWN:
|
|
TrayClick(wparam, TRAYBUTTON_RIGHT);
|
|
break;
|
|
|
|
case WM_RBUTTONDBLCLK:
|
|
TrayDblClick(wparam, TRAYBUTTON_RIGHT);
|
|
break;
|
|
|
|
case WM_MBUTTONDOWN:
|
|
TrayClick(wparam, TRAYBUTTON_MIDDLE);
|
|
break;
|
|
|
|
case WM_MBUTTONDBLCLK:
|
|
TrayDblClick(wparam, TRAYBUTTON_MIDDLE);
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
} else if (nmsg == WM_TASKBARCREATED) {
|
|
AddTrayIcons();
|
|
return 0;
|
|
} else
|
|
return super::WndProc(nmsg, wparam, lparam);
|
|
}
|
|
|
|
virtual void AddTrayIcons() = 0;
|
|
virtual void TrayMouseOver(UINT id) {}
|
|
virtual void TrayClick(UINT id, int btn) {}
|
|
virtual void TrayDblClick(UINT id, int btn) {}
|
|
|
|
protected:
|
|
const UINT WM_TASKBARCREATED;
|
|
};
|
|
|
|
|
|
struct EditController : public SubclassedWindow
|
|
{
|
|
typedef SubclassedWindow super;
|
|
|
|
EditController(HWND hwnd)
|
|
: super(hwnd)
|
|
{
|
|
}
|
|
|
|
protected:
|
|
LRESULT WndProc(UINT nmsg, WPARAM wparam, LPARAM lparam)
|
|
{
|
|
if (nmsg==WM_KEYDOWN && wparam==VK_RETURN) {
|
|
SendParent(WM_COMMAND, MAKEWPARAM(GetDlgCtrlID(_hwnd),1), (LPARAM)_hwnd);
|
|
return 0;
|
|
} else
|
|
return super::WndProc(nmsg, wparam, lparam);
|
|
}
|
|
};
|