Microsoft MVP성태의 닷넷 이야기
COM 개체 관련: 20. 탭 브라우저의 윈도우 핸들 구하기 [링크 복사], [링크+제목 복사],
조회: 32555
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 


탭 브라우저의 윈도우 핸들 구하기


다들 아시는 것처럼, IE 7에서는 "탭 브라우징" 환경을 지원합니다. "일반 사용자" 입장에서는 이는 분명 환영받을 만한 일입니다. 그런데... 개발자 입장에서는 IE 6과 IE 7 간의 환경이 달라짐으로 인해 기존 응용 프로그램들이 정상적으로 구동되지 않는 문제가 있어 ^^ 힘겨울 따름인데요.

오랜만에, IE BHO(Browser Helper Object) 모듈을 다뤄보면서 저도 이 문제에 관심을 갖게 되었습니다. 한번 ^^ 같이 살펴볼까요?




아마도 대부분의 BHO 모듈에서 다음과 같은 식의 SetSite 코드 유형을 구현하고 있을 것입니다.

STDMETHODIMP CHelloWorldBHO::SetSite(IUnknown* pUnkSite)
{
    if (pUnkSite != NULL)
    {
        pUnkSite->QueryInterface(IID_IWebBrowser2, (void**)&m_spWebBrowser);
    }
    else
    {
        m_spWebBrowser.Release();
    }

    return IObjectWithSiteImpl<CHelloWorldBHO>::SetSite(pUnkSite);
}

SetSite의 주된 관심사는 바로 IWebBrowser2 인터페이스 포인터를 얻어내는 것입니다. BHO는 하나의 "Internet Explorer_Server"가 생성될 때마다 그에 상응하는 BHO 개체가 생성되어 연결이 됩니다. 이해를 돕기 위해 다음의 그림을 보겠습니다.

[그림 1: IE 6에서의 웹 브라우저 윈도우 계층]
bho_ie6_ie7_diff_1.png

하나의 IE 인스턴스가 떠 있고, 그와 함께 "Shell DocObject View"와 "Internet Explorer_Server"가 떠 있습니다. 간단하게 생각해서 "Internet Explorer_Server" 윈도우가 생성될 때마다 하나의 BHO가 생성된다고 보시면 됩니다. (그런데, 이것이 "가정"에 불과한 것인지, 어딘가에 씌여져 있는 것을 보고 언제부턴가 그렇게 판단했었던 것인지는 잘 모르겠습니다. 그렇다고 제가 IE 소스를 열어보았던 것도 아니고... ^^; 어쩌면 "IEFrame"과 함께 BHO가 생성되는지도 모를 일입니다.)

음... 그렇다면 "A"라는 웹 사이트를 방문했는데, 그 웹 사이트에서 "팝업 윈도우"를 띄웠다고 가정해보겠습니다. 그럼 몇 개의 BHO 개체가 생성되어져 있을까요? 그렇습니다. 2개입니다. "A" 웹 사이트를 보여주는 IE 인스턴스와 "팝업 윈도우" 역시 IE 인스턴스이기 때문에 총 2개의 BHO 개체가 떠 있는 것입니다. 중요한 것은, 이것들이 단 하나의 "EXE" 프로세스 내에서 생성되기 때문에 경우에 따라서는 static 변수를 쓰는 것이 편리할 수도 있습니다.

"static"을 보자마자 음... "Thread-safe을 고려해야겠군"이라고 생각하신 분이 있을 텐데요. ^^ 훌륭하십니다. 각각의 BHO가 호출되는 스레드를 다음과 같은 식으로 확인해 보시면,

DWORD dwThread = ::GetCurrentThreadId(); 
wchar_t outputDebug[ 4096 ];
wsprintf( outputDebug, L"BHO::SetSite - threadid %d\r\n", dwThread );
::OutputDebugString( outputDebug );

각각 다른 스레드로 활성화되는 것을 볼 수 있습니다. 따라서, static 변수를 사용하실 요량이라면 그에 따른 "동기화 수단"을 제공하시는 것이 좋겠습니다.




IE 6 시절에는, 위와 같은 정도의 지식이라면 충분히 OK일 수 있었습니다. 가만히 보면, 웹 브라우저 윈도우가 생성되면 해당 윈도우에는 각각 다음과 같은 클래스 이름을 가진 윈도우가 하나씩 존재하기 때문입니다.

IEFrame
	Shell DocObject View
		Internet Explorer_Server

위와 같은 간단한 구조로 인해서, 다음과 같은 식으로 윈도우 핸들값을 구하는 것이 일반적이었습니다.

// IEFrame 윈도우 클래스의 윈도우 핸들값
m_spWebBrowser->get_HWND((LONG_PTR*)&m_WebWnd); 

// Internet Explorer_Server의 윈도우 핸들값
HWND ieWindow = ::FindWindowEx( m_WebWnd, L"Internet Explorer_Server", NULL );

음... 위의 결과로 보면, SetSite 메서드가 구현된 BHO가 매핑되는 것은 IEFrame 윈도우 같기도 합니다.

그런데, 이러한 구조가 IE 7에 와서는 다음과 같은 식으로 바뀌었습니다.

[그림 2: IE 7에서의 웹 브라우저 윈도우 계층 - 2개의 웹 사이트 방문]
bho_ie6_ie7_diff_2.png

정리해 보면, 아래와 같은 구조입니다.

** 2개의 웹 사이트 방문시

IEFrame
	TabWindowClass
		Shell DocObject View
			Internet Explorer_Server
	TabWindowClass
		Shell DocObject View
			Internet Explorer_Server

하나의 TabWindowClass가 생성될 때마다 BHO 개체가 하나씩, 별도의 스레드로 호출되어집니다. (음... 여기서 다시 판단해 보면, BHO가 매핑되는 것은 IEFrame이 아닌, 그 하위의 윈도우 중 하나라고 보여집니다.) 실제로 Microsoft에서 공개된 자료를 보아도 탭 윈도우로 구현된 웹 브라우저는, 기존 IE 6의 웹 브라우저를 그대로 하나의 탭으로 옮겼다고 설명하고 있습니다. 이로 인해, 간혹 IE 7이 무겁다고 하는 피드백에 대해서 기존에는 IE 6을 2개를 실행했던 것을 하나의 IE 7프로세스에서 IE 6을 2개를 구동하는 것이므로 하나의 IE 7 프로세스와 하나의 IE 6 프로세스를 비교해서는 안되고, 하나의 IE 7 프로세스와 2개의 IE 6 프로세스를 비교해야 하며, 그렇게 고려해 보면 무거워졌다고 볼 수는 없다는 것입니다.

실제로, "웹 밴드"의 경우를 봐도 IE 6 웹 브라우저 n개를 IE 7 프로세스 하나에서 운영한다는 것이 실감나게 됩니다. 참고로, 아래의 그림 2개는 하나의 IE 7 윈도우에 2개의 탭이 있고, 각각의 탭마다 다르게 보여지는 웹 밴드를 확인할 수 있습니다. 말씀드렸듯이, 별개의 웹 브라우저가 하나의 탭에 할당되고 그 웹 브라우저는 독립적으로 웹 밴드를 소유하기 때문입니다.

bho_ie6_ie7_diff_4.png

bho_ie6_ie7_diff_3.png




자... 이제 다시 다음의 코드를 보면, 분명 문제가 있음을 알 수 있습니다.

// IEFrame 윈도우 클래스의 윈도우 핸들값
m_spWebBrowser->get_HWND((LONG_PTR*)&m_WebWnd); 

// 언제나 첫 번째 Internet Explorer_Server의 윈도우 핸들값 반환
HWND ieWindow = ::FindWindowEx( m_WebWnd, L"Internet Explorer_Server", NULL );

자, 그럼 어떻게 해야 IE 6과 IE 7에서 전혀 코드 변경 없이 원하는 목적을 이룰 수 있을까요? 사실, 그동안은 굳이 필요 없었기 때문에 주목받진 못했지만,,, 이제서야 빛을 보게 되는 코드가 있군요. 바로 다음과 같이 코딩을 해주시면 됩니다.

// IEFrame 윈도우 클래스의 윈도우 핸들값
m_spWebBrowser->get_HWND((LONG_PTR*)&m_WebWnd); 

CComQIPtr<IServiceProvider> pServiceProvider = m_spWebBrowser;
if ( pServiceProvider == NULL )
{
	return S_OK;
}

CComPtr<IOleWindow> pOleWindow;
hr = pServiceProvider->QueryService(SID_SShellBrowser, IID_IOleWindow, (LPVOID *)&pOleWindow);
if ( hr != S_OK )
{
	return S_OK;
}

// Internet Explorer_Server에 해당하는 윈도우 핸들값
hr = pOleWindow->GetWindow( &m_hThisWindow );
if ( hr != S_OK )
{
	return S_OK;
}

에잉... 코드 한번 나오니까 더 설명할 것이 없군요... ^^; IE 6 시절부터 위와 같이 코딩을 해주셨다면 IE 7에 와서 굳이 코드를 변경하실 필요는 없겠지만. FindWindow를 사용해서 코딩을 해주셨다면 반드시 위와 같이 IOleWindow를 사용하도록 바꿔주셔야 합니다.



[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 7/17/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2008-02-12 11시43분
[어쨌건간에] infoscope 뜯다가 SID_SShellBrowser 찾으니 당장에 정 책임님 사이트가 튀어나오는군요. 아니 근데, 왜 Google에 요약된 내용은 전부 영어만?! ㅋ
[guest]
2008-02-13 02시42분
^^
kevin25
2008-09-03 11시41분
크로스 도메인 프레임의 IHTMLDocument을 가져오는 방법
; http://ljh131.tistory.com/96
kevin25

... 121  122  123  124  125  126  127  128  129  130  131  [132]  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
1789정성태10/22/201422509오류 유형: 253. 이벤트 로그 - The client-side extension could not remove user policy settings for '...'
1788정성태10/22/201424351VC++: 82. COM 프로그래밍에서 HRESULT 타입의 S_FALSE는 실패일까요? 성공일까요? [2]
1787정성태10/22/201432603오류 유형: 252. COM 개체 등록시 0x8002801C 오류가 발생한다면?
1786정성태10/22/201434075디버깅 기술: 65. 프로세스 비정상 종료 시 "Debug Diagnostic Tool"를 이용해 덤프를 남기는 방법 [3]파일 다운로드1
1785정성태10/22/201423153오류 유형: 251. 이벤트 로그 - Load control template file /_controltemplates/TaxonomyPicker.ascx failed [1]
1784정성태10/22/201430740.NET Framework: 472. C/C++과 C# 사이의 메모리 할당/해제 방법파일 다운로드1
1783정성태10/21/201424618VC++: 81. 프로그래밍에서 borrowing의 개념
1782정성태10/21/201421395오류 유형: 250. 이벤트 로그 - Application Server job failed for service instance Microsoft.Office.Server.Search.Administration.SearchServiceInstance
1781정성태10/21/201422138디버깅 기술: 64. new/delete의 짝이 맞는 경우에도 메모리 누수가 발생한다면?
1780정성태10/15/201425940오류 유형: 249. The application-specific permission settings do not grant Local Activation permission for the COM Server application with CLSID
1779정성태10/15/201421134오류 유형: 248. Active Directory에서 OU가 지워지지 않는 경우
1778정성태10/10/201419529오류 유형: 247. The Netlogon service could not create server share C:\Windows\SYSVOL\sysvol\[도메인명]\SCRIPTS.
1777정성태10/10/201422563오류 유형: 246. The processing of Group Policy failed. Windows attempted to read the file \\[도메인]\sysvol\[도메인]\Policies\{...GUID...}\gpt.ini
1776정성태10/10/201419618오류 유형: 245. 이벤트 로그 - Name resolution for the name _ldap._tcp.dc._msdcs.[도메인명]. timed out after none of the configured DNS servers responded.
1775정성태10/9/201420856오류 유형: 244. Visual Studio 디버깅 (2) - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
1774정성태10/9/201427774개발 환경 구성: 246. IIS 작업자 프로세스의 20분 자동 재생(Recycle)을 끄는 방법
1773정성태10/8/201431062.NET Framework: 471. 웹 브라우저로 다운로드가 되는 파일을 왜 C# 코드로 하면 안되는 걸까요? [1]
1772정성태10/3/201419870.NET Framework: 470. C# 3.0의 기본 인자(default parameter)가 .NET 1.1/2.0에서도 실행될까? [3]
1771정성태10/2/201428957개발 환경 구성: 245. 실행된 프로세스(EXE)의 명령행 인자를 확인하고 싶다면 - Sysmon [4]
1770정성태10/2/201422822개발 환경 구성: 244. 매크로 정의를 이용해 파일 하나로 C++과 C#에서 공유하는 방법 [1]파일 다운로드1
1769정성태10/1/201425590개발 환경 구성: 243. Scala 개발 환경 구성(JVM, 닷넷) [1]
1768정성태10/1/201420366개발 환경 구성: 242. 배치 파일에서 Thread.Sleep 효과를 주는 방법 [5]
1767정성태10/1/201425763VS.NET IDE: 94. Visual Studio 2012/2013에서의 매크로 구현 - Visual Commander [2]
1766정성태10/1/201423907개발 환경 구성: 241. 책 "프로그래밍 클로저: Lisp"을 읽고 나서. [1]
1765정성태9/30/201427582.NET Framework: 469. Unity3d에서 transform을 변수에 할당해 사용하는 특별한 이유가 있을까요?
1764정성태9/30/201423763오류 유형: 243. 파일 삭제가 안 되는 경우 - The action can't be comleted because the file is open in System
... 121  122  123  124  125  126  127  128  129  130  131  [132]  133  134  135  ...