Microsoft MVP성태의 닷넷 이야기
.NET Framework: 127. ClickOnce로 ActiveX를 같이 배포하는 방법 [링크 복사], [링크+제목 복사],
조회: 39823
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 4개 있습니다.)
.NET Framework: 127. ClickOnce로 ActiveX를 같이 배포하는 방법
; https://www.sysnet.pe.kr/2/0/692

개발 환경 구성: 133. Registry 등록 과정 없이 COM 개체 사용 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1167

개발 환경 구성: 469. Reg-free COM 개체 사용을 위한 manifest 파일 생성 도구 - COMRegFreeManifest
; https://www.sysnet.pe.kr/2/0/12160

개발 환경 구성: 717. Visual Studio - C# 프로젝트에서 레지스트리에 등록하지 않은 COM 개체 참조 및 사용 방법
; https://www.sysnet.pe.kr/2/0/13693





ClickOnce로 ActiveX를 같이 배포하는 방법


이에 대한 주제를 파악하고 나면, 사실 제목이 잘못되었다는 것을 알게 됩니다. 즉, 이 문제는 ClickOnce로 국한되는 것이 아니고, 단지 COM 개체를 레지스트리에 등록하지 않고 사용하는 방법을 설명하는 것이기 때문인데요.

이미 지난 글에서 이에 대한 소개를 잠시 해드렸지요.

Registry 등록 없이 COM 개체 사용 
; https://www.sysnet.pe.kr/2/1/262

ClickOnce에서의 ActiveX 배포도 위의 글에서 설명한 C# 응용 프로그램 범주를 벗어나지 않습니다. 그래도 따라하기 쉽도록... 정리를 한번 해볼까요? ^^




1. WPF 응용 프로그램과 ATL 응용 프로그램을 생성.

테스트를 위해 2개의 프로젝트를 생성합니다. ATL 프로젝트에는 [그림 1]에서 보는 것처럼 "SimpleObject" COM 개체를 추가하고 예를 들기 위한 메서드 하나를 추가합니다.

[그림 1: WPF프로젝트 / ATL프로젝트]
howtodeploy_clickonce_with_com_1.png

=== SimpleObject.cpp ===

STDMETHODIMP CSimpleObject::DoMethod(void)
{
	return S_OK;
}

2. 등록 코드 삭제 / 등록 옵션 해제

UAC 때문에 빌드하면 오류나는 문제도 있고, 어차피 등록이 필요없기 때문에 아예 오류가 발생하지 않도록 등록 코드를 주석 처리하거나, 프로젝트 설정에서 등록 옵션을 해제합니다.

[그림 2: 등록 옵션 제거]
howtodeploy_clickonce_with_com_2.png

3. COM 개체를 위한 manifest 생성

C/C++에서는 해당 COM 개체를 사용하려면 #import 구문을 이용하여 직접 COM 개체의 경로를 지정해서 스마트 포인터로 COM 개체를 사용하는 것이 가능합니다. 반면, C#에서 해당 COM 개체를 사용하려면 어떻게 참조해야 할까요? COM 개체가 등록되지 않았기 때문에 "참조 추가" 대화상자의 "COM" 탭은 무용지물이지요. 대신에 COM 개체에 대한 manifest 파일을 참조하는 것으로 해결할 수 있습니다.

"Registration-Free Activation of COM Components: A Walkthrough"에서 설명하는 것처럼 manifest 파일을 아래와 같이 구성해 주면 됩니다.

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">

  <assemblyIdentity version="1.0.0.1" processorArchitecture="x86" 
     name="AtlTest.dll" type="win32">
  </assemblyIdentity>
  
  <file name="AtlTest.dll">
    <comClass clsid="{1DC804F4-7587-45F2-92C8-7470FE6C091B}" threadingModel="Apartment">
    </comClass>
    <typelib tlbid="{6ECA849F-8500-4539-9287-CFB9D07208A2}" version="1.0" 
             helpdir="AtlTest 1.0 Type Library">
    </typelib>
  </file>
  <comInterfaceExternalProxyStub name="ISimpleObject" 
                                 iid="{9E4A5819-446D-4CEE-ADBB-8D1CE6F1B43A}" 
                                 proxyStubClsid32="{00020424-0000-0000-C000-000000000046}" 
                                 baseInterface="{00000000-0000-0000-C000-000000000046}" 
                                 tlbid="{6ECA849F-8500-4539-9287-CFB9D07208A2}">
  </comInterfaceExternalProxyStub>
</assembly>

위의 manifest 파일에서 사용되는 각각의 값은 .idl 파일과 .rc 파일에서 구할 수 있습니다.

import "oaidl.idl";
import "ocidl.idl";

[
	object,
	uuid(9E4A5819-446D-4CEE-ADBB-8D1CE6F1B43A),
	dual,
	nonextensible,
	helpstring("ISimpleObject Interface"),
	pointer_default(unique)
]
interface ISimpleObject : IDispatch{
	[id(1), helpstring("method DoMethod")] HRESULT DoMethod(void);
};
[
	uuid(6ECA849F-8500-4539-9287-CFB9D07208A2),
	version(1.0),
	helpstring("AtlTest 1.0 Type Library")
]
library AtlTestLib
{
	importlib("stdole2.tlb");
	[
		uuid(1DC804F4-7587-45F2-92C8-7470FE6C091B),
		helpstring("SimpleObject Class")
	]
	coclass SimpleObject
	{
		[default] interface ISimpleObject;
	};
};

[그림 3: assemblyIdentity[@version]으로 사용될 버전]
howtodeploy_clickonce_with_com_3.png

4. WPF 프로젝트에서 manifest 참조 추가


자, 이제 [그림 4]에서 보는 것처럼 "참조 추가"대화상자에서 직접 "AtlTest.dll.manifest" 파일을 지정해서 COM 개체를 참조합니다.

[그림 4: Manifest 파일 참조]
howtodeploy_clickonce_with_com_4.png

정상적으로 참조가 된 경우, [그림 5]에서 보는 것처럼 "References" 노드에 "AtlTest.dll.manifest"가 추가된 것을 볼 수 있습니다.

[그림 5: Manifest 파일 참조된 상태]
howtodeploy_clickonce_with_com_5.png

5. WPF 프로젝트에서 COM 개체 사용 코드 추가

manifest 참조와 함께 Visual Studio는 인텔리센스까지 제공해 주기 때문에 아래와 같이 쉽게 코딩할 수 있습니다.

public partial class Window1 : Window
{
... [중간 생략] ...
    private void Window_Loaded(object sender, RoutedEventArgs e)
    {
        AtlTestLib.SimpleObjectClass soc = new AtlTestLib.SimpleObjectClass();
        soc.DoMethod();
    }
}

이렇게 해서 최종 빌드하고 나면, WPF 빌드 출력 결과물은 [그림 6]과 같이 됩니다.

[그림 6: WPF 응용 프로그램 빌드 후 결과물]
howtodeploy_clickonce_with_com_6.png

WpfApplication1.exe.manifest가 눈에 띄죠. 그런데, 아쉬운 것은 "WpfApplication1.vshost.exe.manifest" 파일이 없다는 점입니다. 이것이 뭘 의미하냐면? 이 상태에서 "WpfApplication1.exe"를 실행시키면 오류가 발생하지 않지만, Visual Studio에서 "F5" 디버깅을 하는 경우에는 오류가 발생한다는 문제가 있습니다. (아마도 이것은 Visual Studio에서 Hosting Process(vshost.exe)의 manifest에 대한 관리적인 측면에 오작동이 있는 것 같은데,... VS2010에서는 개선이 되길 바랍니다.)

그래서, "F5 디버깅"을 하기 위해서는 아래의 2가지 방법 중 하나를 선택해야 합니다.

  • 프로젝트 설정에서 "Enable the Visual Studio hosting process" 옵션을 해제한다.
  • 개발자 PC에서는 아예 COM 개체를 등록해 둔다.

뭐... 어쩔 수가 없겠지요.
이렇게 해서 ClickOnce로 배포하고 실행해 보면 정상적으로 COM 개체의 메서드가 실행되는 것을 확인할 수 있습니다. 참고로, 아래는 ClickOnce로 배포된 경우의 클라이언트 측 폴더 내용입니다.

[그림 7: ClickOnce로 배포된 폴더의 내용]
howtodeploy_clickonce_with_com_7.png

보시는 것처럼 정상적으로 AtlTest.dll 파일이 내려가 있습니다.

첨부된 파일은 이러한 과정이 적용된 WPF와 ATL프로젝트를 담고 있는 솔루션입니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/21/2024]

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

비밀번호

댓글 작성자
 



2009-04-21 05시49분
Deployment in Visual Studio
- Deploying COM Components with ClickOnce
; https://learn.microsoft.com/en-us/previous-versions/visualstudio/visual-studio-2015/deployment/deploying-com-components-with-clickonce

위의 토픽에서도 자세히 다루고 있군요. ^^
kevin25
2009-06-20 12시24분
vshost.exe에 대한 manifest 지원이 안 되는 문제로 인해 "Enable the Visual Studio hosting process" 옵션을 해제하는 것은 Connect에 피드백을 올렸는데... 의외로 마이크로소프트에 이에 대한 문제를 보고 받은 적이 없나 봅니다. ^^;

Using registration-free COM components in vshost.exe is not possible
; (broken) http://connect.microsoft.com/VisualStudio/feedback/ViewFeedback.aspx?FeedbackID=467551

어쨌든... 수정이 되면 좋겠군요. ^^
kevin25

... 61  62  63  64  65  66  67  68  69  70  71  72  73  74  [75]  ...
NoWriterDateCnt.TitleFile(s)
12153정성태2/23/202024419.NET Framework: 898. Trampoline을 이용한 후킹의 한계파일 다운로드1
12152정성태2/23/202021422.NET Framework: 897. 실행 시에 메서드 가로채기 - CLR Injection: Runtime Method Replacer 개선 - 세 번째 이야기(Trampoline 후킹)파일 다운로드1
12151정성태2/22/202024058.NET Framework: 896. C# - Win32 API를 Trampoline 기법을 이용해 C# 메서드로 가로채는 방법 - 두 번째 이야기 (원본 함수 호출)파일 다운로드1
12150정성태2/21/202024148.NET Framework: 895. C# - Win32 API를 Trampoline 기법을 이용해 C# 메서드로 가로채는 방법 [1]파일 다운로드1
12149정성태2/20/202021056.NET Framework: 894. eBEST C# XingAPI 래퍼 - 연속 조회 처리 방법 [1]
12148정성태2/19/202025734디버깅 기술: 163. x64 환경에서 구현하는 다양한 Trampoline 기법 [1]
12147정성태2/19/202021044디버깅 기술: 162. x86/x64의 기계어 코드 최대 길이
12146정성태2/18/202022234.NET Framework: 893. eBEST C# XingAPI 래퍼 - 로그인 처리파일 다운로드1
12145정성태2/18/202023851.NET Framework: 892. eBEST C# XingAPI 래퍼 - Sqlite 지원 추가파일 다운로드1
12144정성태2/13/202024026.NET Framework: 891. 실행 시에 메서드 가로채기 - CLR Injection: Runtime Method Replacer 개선 - 두 번째 이야기파일 다운로드1
12143정성태2/13/202018443.NET Framework: 890. 상황별 GetFunctionPointer 반환값 정리 - x64파일 다운로드1
12142정성태2/12/202022357.NET Framework: 889. C# 코드로 접근하는 MethodDesc, MethodTable파일 다운로드1
12141정성태2/10/202021369.NET Framework: 888. C# - ASP.NET Core 웹 응용 프로그램의 출력 가로채기 [2]파일 다운로드1
12140정성태2/10/202022715.NET Framework: 887. C# - ASP.NET 웹 응용 프로그램의 출력 가로채기파일 다운로드1
12139정성태2/9/202022409.NET Framework: 886. C# - Console 응용 프로그램에서 UI 스레드 구현 방법
12138정성태2/9/202028616.NET Framework: 885. C# - 닷넷 응용 프로그램에서 SQLite 사용 [6]파일 다운로드1
12137정성태2/9/202020261오류 유형: 592. [AhnLab] 경고 - 디버거 실행을 탐지했습니다.
12136정성태2/6/202021907Windows: 168. Windows + S(또는 Q)로 뜨는 작업 표시줄의 검색 바가 동작하지 않는 경우
12135정성태2/6/202027707개발 환경 구성: 468. Nuget 패키지의 로컬 보관 폴더를 옮기는 방법 [2]
12134정성태2/5/202024974.NET Framework: 884. eBEST XingAPI의 C# 래퍼 버전 - XingAPINet Nuget 패키지 [5]파일 다운로드1
12133정성태2/5/202022725디버깅 기술: 161. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 - 두 번째 이야기
12132정성태1/28/202025741.NET Framework: 883. C#으로 구현하는 Win32 API 후킹(예: Sleep 호출 가로채기) [1]파일 다운로드1
12131정성태1/27/202024473개발 환경 구성: 467. LocaleEmulator를 이용해 유니코드를 지원하지 않는(한글이 깨지는) 프로그램을 실행하는 방법 [1]
12130정성태1/26/202022033VS.NET IDE: 142. Visual Studio에서 windbg의 "Open Executable..."처럼 EXE를 직접 열어 디버깅을 시작하는 방법
12129정성태1/26/202029062.NET Framework: 882. C# - 키움 Open API+ 사용 시 Registry 등록 없이 KHOpenAPI.ocx 사용하는 방법 [3]
12128정성태1/26/202023173오류 유형: 591. The code execution cannot proceed because mfc100.dll was not found. Reinstalling the program may fix this problem.
... 61  62  63  64  65  66  67  68  69  70  71  72  73  74  [75]  ...