Microsoft MVP성태의 닷넷 이야기
.NET Framework: 370. C# - WebKit .NET 사용 [링크 복사], [링크+제목 복사],
조회: 35514
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)

C# - WebKit.NET 사용

일반적으로 Microsoft 플랫폼에서는 Internet Explorer 프로그램이 ActiveX로 제공되는 WebBrowser 컨트롤을 사용하게 되는데요. Web Renderer로서 오픈소스인 WebKit도 선택사항일 수 있습니다.

게다가 C#이 워낙 interop 기능이 훌륭해서 WebKit을 닷넷 응용 프로그램에서 사용하는 것도 가능한데요. 다행히 이것조차도 우리가 만들 필요 없이 이미 다음의 사이트에 공개되어 있습니다. ^^

WebKit .NET
; http://webkitdotnet.sourceforge.net/downloads.php

WebKit .NET 소스 코드는 WebKit이 컴파일 된 상태에서 WebKit.Interop.dll만으로 연동하는 형식을 취하고 있습니다. 따라서, 그냥 소스 코드를 다운로드해도 그다지 어렵지 않게 빌드할 수 있습니다.

그래서, 이 글에서는 WebKit .NET 소스 코드를 다운로드한 것으로 진행합니다.

압축 해제를 하면 WebKit.NET.sln 파일과 이하 프로젝트 파일들이 나오고 Visual Studio 2012에서도 WebKit.NET.sln 파일을 그냥 로드할 수 있습니다. 그럼 다음과 같은 2개의 프로젝트를 볼 수 있는데요.

  1. WebKitBrowser
  2. WebKitBrowserTest

WebKitBrowser 프로젝트가 WebKit을 WinForm UserControl로 노출시켜주는 역할을 하고, WebKitBrowserTest 프로젝트는 WebKitBrowser 사용자 컨트롤을 이용해 만든 간단한 웹 브라우저입니다.

그런데 빌드해 보면 약간 오류가 발생합니다. 우선, WebKit.interop.dll이 .NET 4.0으로 빌드된 것이기 때문에 각각 로드된 2개의 프로젝트 모두 Target Framework를 .NET Framework 4.0으로 바꿔줍니다.

그다음, 몇몇 소스 코드에서 webkit.interop.dll에는 구현 클래스를 인자로 받도록 되어 있지만 정작 인터페이스를 받도록 구현된 소스 코드들이 있는데요. 빌드 시 WebPolicyDelegate.cs에서 가장 먼저 오류가 발생하는데, 구현 함수의 signature를 각각 다음과 같이 변경해 주어야 합니다.

소스 코드: WebPolicyDelegate.cs

public void decidePolicyForMIMEType(WebView WebView, string type, IWebURLRequest request, webFrame frame, IWebPolicyDecisionListener listener)

==> public void decidePolicyForMIMEType(WebView WebView, string type, WebURLRequest request, webFrame frame, IWebPolicyDecisionListener listener)


public void decidePolicyForNavigationAction(WebView WebView, CFDictionaryPropertyBag actionInformation, IWebURLRequest request, webFrame frame, IWebPolicyDecisionListener listener)

==> public void decidePolicyForNavigationAction(WebView WebView, CFDictionaryPropertyBag actionInformation, WebURLRequest request, webFrame frame, IWebPolicyDecisionListener listener)
  
  
public void decidePolicyForNewWindowAction(WebView WebView, CFDictionaryPropertyBag actionInformation, IWebURLRequest request, string frameName, IWebPolicyDecisionListener listener)
    
==> public void decidePolicyForNewWindowAction(WebView WebView, CFDictionaryPropertyBag actionInformation, WebURLRequest request, string frameName, IWebPolicyDecisionListener listener)

이와 유사한 오류가 WebResourceLoadDelegate.cs, WebUIDelegate.cs 파일에도 발생하는데요. 역시 같은 방식으로 인터페이스 형식을 구현 클래스 형식으로 바꿔주는 식으로 해결하면 됩니다.

마지막으로 오류 하나가 WebKitBrowser.cs에서 발생하는데,

public void Navigate(string url)
{
    if (loaded)
    {
        // prepend with "http://" if url not well formed
        if (!Uri.IsWellFormedUriString(url, UriKind.Absolute))
            url = "http://" + url;

        activationContext.Activate();

        WebMutableURLRequest request = new WebMutableURLRequestClass();
        request.initWithURL(url, _WebURLRequestCachePolicy.WebURLRequestUseProtocolCachePolicy, 60);
        request.setHTTPMethod("GET");

        webView.mainFrame().loadRequest(request); // 컴파일 오류!

        activationContext.Deactivate();
    }
    else
    {
        initialUrl = url.Length == 0 ? null : new Uri(url);
    }
}

왜냐하면, loadRequest의 첫 번째 인자는 WebURLRequest 형식을 받는데, WebMutableURLRequest는 WebURLRequest 타입 상속을 받지 않고 IWebURLRequest 인터페이스 상속을 받으므로 사실상 2개는 암시적 변환에 위배되므로 그와 같은 오류가 발생하는 것입니다.

따라서, 다음과 같이 적당한 변환을 해주면 컴파일도 되고 실행도 정상적으로 됩니다.

WebMutableURLRequest request = new WebMutableURLRequestClass();
request.initWithURL(url, _WebURLRequestCachePolicy.WebURLRequestUseProtocolCachePolicy, 60);
request.setHTTPMethod("GET");

IWebURLRequest req = request as IWebURLRequest;
WebURLRequest req2 = req as WebURLRequest;

webView.mainFrame().loadRequest(req2);

복잡한가요? ^^ 그래서 제가 첨부 파일에 위의 변경 사항을 모두 적용한 Visual Studio 2012용 프로젝트를 올려 두었으니 그냥 다운로드해 Ctrl + F5 키로 실행만 해주시면 됩니다. 그럼 다음과 같이 예제 웹 브라우저가 실행되는 것을 볼 수 있습니다.

webkit_browser_1.png

대강의 사용법은 WebKitBrowserTest 프로젝트의 소스 코드를 참고해도 되지만 다음의 글에서도 자세하게 설명하고 있으니 참고하시면 됩니다.

Tutorial - The Basics
; http://webkitdotnet.sourceforge.net/basics.php

참고로, Visual Studio에서 F5 디버깅을 하면 몇몇 NotImplementedException에서 멈출 수가 있습니다. 이것은 의도된 것이니 그냥 디버깅을 계속하면 정상적으로 실행되므로 무시하시면 됩니다.




와~~~ 그래서 기쁜 마음으로, 우리 회사 제품인 Jennifer .NET의 모니터링 화면을 보려고 했는데요. ^^ 현재 제니퍼 4.5 버전은 Java Applet을 사용하고 있는데 WebKitBrowserTest 프로그램으로 방문하니 ^^; 비정상 종료가 되어 버립니다. 자바의 hs_err_pid...log 파일이 남는 걸로 봐서 JVM과의 충돌이 발생한 듯 싶은데요. 이것이 .NET 층이 생겨서 그런 것인지, WebKit .NET에 포함된 WebKit의 문제인지는 판단할 수가 없었습니다. (테스트해 보니, JRE 6/7 모두 비정상 종료!)

결국, Applet이나 ActiveX가 없는 제한된 상황에서나 WebKit.NET이 선택 옵션으로 들어갈 것 같습니다. ^^




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/26/2021]

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

비밀번호

댓글 작성자
 



2013-06-06 11시51분
[짜두] 윈포에서 다른 브라우저를 쓴다는 생각이 전환이 대단해보입니다~ㅎ 멋지다는~ ^^
[guest]
2013-06-08 02시25분
요거 말고, chromium을 기반으로 한 웹 브라우저 컨트롤도 있어요. 조만간 소개할께요. ^^
정성태

1  2  3  4  5  [6]  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13829정성태11/25/20246690스크립트: 67. 파이썬 - Windows 버전에서 함께 설치되는 py.exe
13828정성태11/25/20245180개발 환경 구성: 735. Azure - 압축 파일을 이용한 web app 배포 시 디렉터리 구분이 안 되는 문제파일 다운로드1
13827정성태11/25/20246010Windows: 273. Windows 환경의 파일 압축 방법 (tar, Compress-Archive)
13826정성태11/21/20246388닷넷: 2313. C# - (비밀번호 등의) Console로부터 입력받을 때 문자열 출력 숨기기(echo 끄기)파일 다운로드1
13825정성태11/21/20247044Linux: 110. eBPF / bpf2go - BPF_RINGBUF_OUTPUT / BPF_MAP_TYPE_RINGBUF 사용법
13824정성태11/20/20245425Linux: 109. eBPF / bpf2go - BPF_PERF_OUTPUT / BPF_MAP_TYPE_PERF_EVENT_ARRAY 사용법
13823정성태11/20/20246614개발 환경 구성: 734. Ubuntu에 docker, kubernetes (k3s) 설치
13822정성태11/20/20246493개발 환경 구성: 733. Windbg - VirtualBox VM의 커널 디버거 연결 시 COM 포트가 없는 경우
13821정성태11/18/20246094Linux: 108. Linux와 Windows의 프로세스/스레드 ID 관리 방식
13820정성태11/18/20246551VS.NET IDE: 195. Visual C++ - C# 프로젝트처럼 CopyToOutputDirectory 항목을 추가하는 방법
13819정성태11/15/20245128Linux: 107. eBPF - libbpf CO-RE의 CONFIG_DEBUG_INFO_BTF 빌드 여부에 대한 의존성
13818정성태11/15/20246672Windows: 272. Windows 11 24H2 - sudo 추가
13817정성태11/14/20245895Linux: 106. eBPF / bpf2go - (BPF_MAP_TYPE_HASH) Map을 이용한 전역 변수 구현
13816정성태11/14/20246851닷넷: 2312. C#, C++ - Windows / Linux 환경의 Thread Name 설정파일 다운로드1
13815정성태11/13/20245468Linux: 105. eBPF - bpf2go에서 전역 변수 설정 방법
13814정성태11/13/20246110닷넷: 2311. C# - Windows / Linux 환경에서 Native Thread ID 가져오기파일 다운로드1
13813정성태11/12/20246654닷넷: 2310. .NET의 Rune 타입과 emoji 표현파일 다운로드1
13812정성태11/11/202410246오류 유형: 933. Active Directory - The forest functional level is not supported.
13811정성태11/11/20245849Linux: 104. Linux - COLUMNS 환경변수가 언제나 80으로 설정되는 환경
13810정성태11/10/20246862Linux: 103. eBPF (bpf2go) - Tracepoint를 이용한 트레이스 (BPF_PROG_TYPE_TRACEPOINT)
13809정성태11/10/20246496Windows: 271. 윈도우 서버 2025 마이그레이션
13808정성태11/9/20246805오류 유형: 932. Linux - 커널 업그레이드 후 "error: bad shim signature" 오류 발생
13807정성태11/9/20245680Linux: 102. Linux - 커널 이미지 파일 서명 (Ubuntu 환경)
13806정성태11/8/20245861Windows: 270. 어댑터 상세 정보(Network Connection Details) 창의 내용이 비어 있는 경우
13805정성태11/8/20245470오류 유형: 931. Active Directory의 adprep 또는 복제가 안 되는 경우
13804정성태11/7/20247043Linux: 101. eBPF 함수의 인자를 다루는 방법
1  2  3  4  5  [6]  7  8  9  10  11  12  13  14  15  ...