볼랜드포럼에 올린글을 블로그에 재게시 합니다.
http://cbuilder.borlandforum.com/impboard/impboard.dll?action=read&db=bcb_tutorial&no=132
http://cbuilder.borlandforum.com/impboard/impboard.dll?action=read&db=bcb_tutorial&no=132
윈도우는 사용자로부터 마우스 , 키보드 등으로부터 입력을 받아 메시지를 발생시키고 이 메시지를 처리하는 함수를
실행시키는 방식으로 작동합니다. 메시지를 처리하는 함수를 메시지 핸들러라고 합니다.
C++ Builder 에서 구현하는 방법은 3가지가 있습니다.
(1) WndProc 함수사용
(2) Dispatch 함수 오버라이딩
(3) (2) 를 MESSAGE_MAP macro 로 구현
(2) 와 (3) 은 같습니다. 모양만 다를 뿐입니다.
이중에서 (2) 번의 예는 김태선님께서 올리신 자료를 참고하세요.
KTS Compo
여기서는 (3) 번의 방법으로 구현하는 방법을 알아보겠습니다.
다음과 같이 구현합니다. OnMouseEnter , OnMouseLeave property 를 __published 섹션에 넣습니다. 이벤트도 property 이니까요
class PACKAGE TMyImage : public TImage
{
private:
TNotifyEvent FOnMouseEnter;
TNotifyEvent FOnMouseLeave;
MESSAGE void __fastcall CMMouseEnter(TMessage &Message); //MouseEnter 메시지 핸들러
MESSAGE void __fastcall CMMouseLeave(TMessage &Message); //MouseLeave 메시지 핸들러
protected:
BEGIN_MESSAGE_MAP
VCL_MESSAGE_HANDLER(CM_MOUSEENTER, TMessage, CMMouseEnter)
VCL_MESSAGE_HANDLER(CM_MOUSELEAVE, TMessage, CMMouseLeave)
END_MESSAGE_MAP(TImage)
public:
__fastcall TMyImage(TComponent* Owner);
__published:
__property TNotifyEvent OnMouseEnter = {read = FOnMouseEnter, write = FOnMouseEnter};
__property TNotifyEvent OnMouseLeave = {read = FOnMouseLeave, write = FOnMouseLeave};
};
{
private:
TNotifyEvent FOnMouseEnter;
TNotifyEvent FOnMouseLeave;
MESSAGE void __fastcall CMMouseEnter(TMessage &Message); //MouseEnter 메시지 핸들러
MESSAGE void __fastcall CMMouseLeave(TMessage &Message); //MouseLeave 메시지 핸들러
protected:
BEGIN_MESSAGE_MAP
VCL_MESSAGE_HANDLER(CM_MOUSEENTER, TMessage, CMMouseEnter)
VCL_MESSAGE_HANDLER(CM_MOUSELEAVE, TMessage, CMMouseLeave)
END_MESSAGE_MAP(TImage)
public:
__fastcall TMyImage(TComponent* Owner);
__published:
__property TNotifyEvent OnMouseEnter = {read = FOnMouseEnter, write = FOnMouseEnter};
__property TNotifyEvent OnMouseLeave = {read = FOnMouseLeave, write = FOnMouseLeave};
};
Dispatch 함수를 직접 사용하지 않고 Macro 를 이용하여 메시지 핸들러와 메시지가 어떻게 연결되어 있는지 알기
쉽도록 한 것입니다. 하지만 이것 또한 내부적으로는 Dispatch 함수와 switch - case , 그리고 마지막은 기반 클래스의
Dispatch 를 호출하는 것으로 되어 있습니다. MESSAGE 는 아무의미없는 Macro 입니다. 그냥 Message Handler 라는 것을 강조하기 위해서 넣은 것입니다. 스타일에 따라서 안 넣는 분들도 많으니..
메시지와 메시지 핸들러를 연결하는 형식은 VCL_MESSAGE_HANDLER(처리할 메시지, 메시지 타입, 메시지 핸들러) 입니다.
VCL_MESSAGE_HANDLER(CM_MOUSEENTER, TMessage, CMMouseEnter)
CM_MOUSEENTER 는 Control Message 입니다. CM_ 으로 시작합니다. VCL 자체 메시지입니다.
CMMouseEnter 는 메시지 핸들러입니다. 밑줄 _ 를 제거하고 대소문자 구분을 해줍니다. 그냥 VCL 관습입니다.
sysmac.h 에 있는 MESSAGE_MAP macro 부분
#define BEGIN_MESSAGE_MAP virtual void __fastcall Dispatch(void *Message) \
{ \
switch (((PMessage)Message)->Msg) \
{
#define VCL_MESSAGE_HANDLER(msg,type,meth) \
case msg: \
meth(*((type *)Message)); \
break;
#define END_MESSAGE_MAP(base) default: \
base::Dispatch(Message); \
break; \
} \
}
{ \
switch (((PMessage)Message)->Msg) \
{
#define VCL_MESSAGE_HANDLER(msg,type,meth) \
case msg: \
meth(*((type *)Message)); \
break;
#define END_MESSAGE_MAP(base) default: \
base::Dispatch(Message); \
break; \
} \
}
이벤트를 처리하는 과정은 MainWndProc 이 WndProc 을 호출하고 Dispatch 를 호출해서 만약 사용자가 이벤트 핸들러를
작성했다면 이벤트 핸들러를 호출하고 아니면 기본 핸들러를 호출합니다. 메시지 핸들러가 이벤트 핸들러를 호출하는 방법은 다음과 같습니다.
void __fastcall TMyImage::CMMouseEnter(TMessage &Message)
{
if(FOnMouseEnter) //사용자가 작성한 OnMouseEnter Event Handler 가 있다면
FOnMouseEnter(this); //OnMouseEnter Event Handler 호출
}
{
if(FOnMouseEnter) //사용자가 작성한 OnMouseEnter Event Handler 가 있다면
FOnMouseEnter(this); //OnMouseEnter Event Handler 호출
}
FOnMouseEnter 는 TNotifyEvent 형입니다.
TNotifyEvent FOnMouseEnter;
TNotifyEvent 형은 사용자가 작성한 이벤트 핸들러를 호출하는데 사용되는 함수 포인터형입니다. 여기서 한가지 신기한 점이 생깁니다. C++Builder 를 이용하여 폼에 Button을 올려놓고 TButton 타입의 객체 Button1 의 Event Handler 를 작성하면 Form1의 멤버함수가 됩니다. Button1을 Click 했을때의 작동을 결정하는 Button1Click 함수는 TButton class 의 멤버 함수가 되어야 하는 것이 아닐까요? 객체 지향 프로그래밍에서 객체의 동작은 해당 객체의 class 에서 멤버함수로 등록되어야 합니다. 그런데 엉뚱하게도 Form1의 멤버함수로 등록됩니다. 이것이 C++Builder 의 특징입니다. 기능을 추가했을때 해당 class 를 수정 하는 대신, 추가된 기능을 함수 포인터로 연결해 줍니다. 그리고 Form 위에 올려놓은 컴포넌트들의 동작을 모두 한개의 소스로 작성합니다. 이점이 C++Builder 를 처음 접했을때 당황스러운 점입니다. class 좀 배우고 객체 지향 프로그래밍이 뭔지 알듯말듯한데 배운것과는 약간 다른 방법을 사용하니까요.
Classes.hpp TNotifyEvent 선언 부분
typedef void __fastcall (__closure *TNotifyEvent)(System::TObject* Sender);
특이한 것은 __closure 입니다. 이것은 8바이트 포인터로 앞에는 함수의 주소, 뒤에는 함수가 있는 클래스의 인스턴스라고... 합니다. 자세한 것은 C++Builder 의 C++ 확장 을 참조하세요. Button1 과 이벤트 핸들러인 Button1Click 함수가 어떻게 연결되는지 눈치로 알 수 있습니다. 뒤에는 인자를 어떻게 넘길것인가를 나타냅니다. TNotifyEvent 형은 기본형으로 TObject* Sender 를 인자로 넘겨줍니다.
Delphi 에서는 다음과 같이 선언합니다. of object 를 사용하는 군요.
Classes.pas TNotifyEvent 선언 부분
TNotifyEvent = procedure(Sender: TObject) of object;
이것도 참고하세요.
컴포넌트 만들기.. 속성, 메소드, 이벤트
TImage , TMyImage 의 Event 탭을 비교하면 다음과 같습니다.
C++Builder 로 된 메시지 핸들러 부분을 Delphi 로 표현한다면 다음과 같습니다.
interface 부분
procedure CMMouseEnter(var Message: TMessage); message CM_MOUSEENTER;
procedure CMMouseLeave(var Message: TMessage); message CM_MOUSELEAVE;
procedure CMMouseLeave(var Message: TMessage); message CM_MOUSELEAVE;
implementation 부분
procedure TMyImage.CMMouseEnter(var Message: TMessage);
begin
if Assigned(FOnMouseEnter) then
FOnMouseEnter(Self);
end;
procedure TMyImage.CMMouseLeave(var Message: TMessage);
begin
if Assigned(FOnMouseLeave) then
FOnMouseLeave(Self);
end;
begin
if Assigned(FOnMouseEnter) then
FOnMouseEnter(Self);
end;
procedure TMyImage.CMMouseLeave(var Message: TMessage);
begin
if Assigned(FOnMouseLeave) then
FOnMouseLeave(Self);
end;
MESSAGE_MAP Macro 를 사용하는 대신 message 키워드를 사용하는 것을 제외하면 C++Builder 와 유사합니다.
'기술탐구' 카테고리의 다른 글
[C++Builder] TApLabel 만들기 (2) Object 형 property (0) | 2008.07.23 |
---|---|
[C++Builder] TApLabel 만들기 (1) 준비 (2) | 2008.07.23 |
[C++Builder] About Property 를 추가한 TMyComponent (0) | 2008.07.23 |
[C++Builder] IDE 확장 - 나만의 Component Editor 를 추가해보자 (0) | 2008.07.23 |
[C++Builder] Component Editor 가 추가된 TMyLabel (0) | 2008.07.23 |