Microsoft MVP성태의 닷넷 이야기
OCX 로드 관련 질문입니다. [링크 복사], [링크+제목 복사]
조회: 8422
글쓴 사람
링크의 전설
홈페이지
첨부 파일
안녕하세요.


현재 진행중인 프로젝트에서 ActiveX 컨트롤을 동적으로 로드하여 사용해야하는데
구글링(현재 COM 관련 지식이 전무한 상태)해봐도 적절한 해결책을 찾을 수 없어 질문드립니다.


최종 구현 목표
1. .Net 환경에서 동적(n개 ocx를 사용자 선택에 따라)으로 OCX를 로드하여 윈폼에 올림
2. 스크립트(현재는 ClearScript)에서 로드한 OCX 객체의 Property, Method, Event 접근하여 관련 작업을 수행
   ex) axHost.NextDay(); // 스크립트에서 이와 같이 접근하여 사용


step1
아래와 같이 AxHost 상속하여 OCX 로드

class AxHostEx : AxHost
{
     public AxHostEx(string strCLSID)
      : base(strCLSID)
      {
      }

     // 아무런 추가 구현 없음
}

결과 : 로드되고 폼에 올려지며 정상 동작하나 해당 OCX 고유의 Property, Method, Event에 .Net 인터페이스로 접근 불가
       (해보기 전까진 어떤 형태로든 '.Net 님이 다 알아서 해주실꺼야'라고 생각하고 있었습니다)

상태 : 이후 조사해 본바로는 AxHost를 상속받은 클래스에 COM객체와 직접 통신는 코드(기계적?)를 작성하여 해당 멤버들을 직접 구현해야 한다는 것을 알게되었습니다(제대로 이해한 것이 맞다면).
       COM 관련 지식이 전무한 상태라 관련 내용을 살펴봐도 잘 이해가되지 않고
       무엇보다 코드를 직접 구현하고 컴파일해야하는 정적인 방식이라 최종 구현 목표에 부합되지 않아 보류

step2
AxImp.exe 사용 COM Interop 어셈블리와 COM 프록시 어셈블리를 생성하여 어셈블리 로드

결과 : 완전하지 않지만 구현 목표에 부합하며 잘 동작함

상태 : 문제없이 잘 동작 하지만 동작(구현) 방식이 우아(?)하지도 안전(?)하지도 않는데다
       배포되어 사용되는 시점에 모양새가 안타까워 가능한 다른 방법으로 찾고 있음


질문 요약
.Net에서 AxImp.exe를 사용하지 않고 동적으로 로드한 OCX의 인터페이스(Property, Method, Event)를 .Net 인터페이스 형태로 제공하는 것이 가능한가?
(step1의 구현 방식에서 로드한 COM 개체를 분석해 동적으로 멤버를 생성/추가하면되겠다(리플렉션이든 뭐든?)라고 막연히 생각해보는데 이것이 기술적으로 가능한 것인지?
만약 그런 방식으로 구현이 가능하다면 AxImp.exe와 같은 도구를 별도로 제공할필요가 없었텐데하는 의문에 질문하게 되었습니다)

혹은 상기 최종 구현 목표를 달성을 위한 또 다른 방법이 있다면 조언 부탁합니다.



답변 부탁드립니다.

감사합니다.


※ 첨부한 테스트 프로젝트는 용량 관계상 스크립트 라이브러리 패키지(Nuget)를 삭제한 상태입니다.
   OCX 로드 테스트에 사용한 MSCAL.OCX도 첨부했습니다.
   개발환경 : Win7 32bit, VS2015




donaricano-btn



[최초 등록일: ]
[최종 수정일: 10/12/2015 ]


비밀번호

댓글 쓴 사람
 



2015-10-12 12시44분
결국, AxImp가 작성해 주는 것은 모든 ActiveX에 대한 공통적인 코드일 뿐입니다. 따라서, clsid나 progid를 넘겨주는 부분을 설정할 수 있도록 하기만 하면 AxImp가 작성해 준 코드를 그대로 재사용할 수 있습니다. 즉, aximp.exe를 같이 배포하지 않아도 됩니다.
정성태
2015-10-13 07시26분
[링크의 전설] 답변 감사합니다.

답변해주신 내용중에 "AxImp가 작성해 준 코드를 그대로 재사용할 수 있습니다"가 구체적으로 어떤 내용인지 잘 이해되지 않습니다.


우선 제가 테스트 한 내용(step2)을 좀더 자세히 설명하면
1. 첨부 파일에 첨부한 MSCAL.OCX를 아래와 같은 명령으로 변환했습니다(예제는 MSCAL.OCX로 한정하여 작성했지만 실제로 대상 OCX는 런타임에 사용자가 결정).
aximp.exe mscal.ocx

변환 결과 아래 두 파일이 생성됩니다.

AxMSACAL.dll(윈폼 래퍼(프록시))
MSACAL.dll(RCW)

2. .Net 클라이언트에서는 AxMSACAL.dll 로드 후 clsid로 Type(AxCalendar)을 객체를 생성/사용했습니다.
   (.Net 클라이언트에서 macal.ocx의 고유의 모든 PME에 .Net 형식의 인터페이스로 접근)

코드를 그대로 재사용하라는 의미가 위의 MSACAL.dll 코드를 참조하여 AxHost를 상속받아 적절하게 구현하여 사용하라는 말씀인가요?
아니면 다른 의미인가요?

제가 질문드린 내용을 다시 한번 정리하면
프로그램 런타임에 특정 ocx가 주어졌을때 aximp.exe로 변환하여 생성되는 윈폼래퍼(위 예에서 AxMSACAL.dll)가 제공하는 수준의 타입(정보(PME))을
aximp.exe를 사용하지 않고
프로그래밍적인 방법으로 클라이언트에게 제공하는 것이 가능한가? 였습니다. 답변해주신 내용이 핀트가 어긋난것은 아닌지 확인차 댓글 달아봅니다.
[손님]
2015-10-13 08시29분
[spowner] 정태님 말은
로딩하고자 하는 n개의 ocx들이 모두 동일한 인터페이스(Property, Method, Event)를 가지고 있는 경우 AxImp를 이용해 한번만 레퍼 클레스를 생성한 후, clsid나 progid를 넘겨주는 식으로 설정만 할 수 있다면
"동일한 인터페이스"로 각각의 ocx에 aximp.exe 없이 사용가능하다는 얘기로 이해합니다.

어쩌면 AxHost클래스를 이용해 각각의 Property, Method, Event에 대한 레핑을 만든 후 사용한다라면 AxHostEx로도 사용 가능할 것 같은데요..
만약, 각각의 ocx에서 제공하는 Property, Method, Event가 동일한 인터페이스가 아니라면 역시 AxHost에 관련 호출 가능한 invoke 기능이 제공할 것도 같습니다.

한번 첨부파일을 받아 저도 한번 테스트를 해봐야겠습니다 하하
[손님]
2015-10-13 11시37분
아... 제가 실수했습니다. 부분적으로만 보고 답변을 드렸군요. 원하시는 것이 로드뿐만 아니라 해당 ocx에 포함된 멤버(Method, Event...)도 동적으로 알고 싶다고 하셨는데, 이것이 가능하려면 COM의 TypeLibrary를 해석해야 합니다. (TypeLibrary는 닷넷의 메타데이터라고 보시면 됩니다.) 다행히 이런 기능을 이미 mscorlib 어셈블리에 TypeLibConverter.ConvertTypeLibToAssembly라는 메서드로 제공하고 있는데요. 사실 AxImp.exe도 .NET Reflector로 보면 결국 저 메서드를 내부적으로 호출하고 있습니다.

자세한 것은 제 설명보다는 AxImp.exe를 .NET Reflector로 보시고 필요한 코드만 가져가 사용하시면 될 듯 합니다.

하지만, 차라리 그러느니 그냥 AxImp.exe를 사용하시는 것이 속편할지도 모릅니다. 첨부하신 소스코드에 보면 "간헐적으로 에러가 난다"고 했는데요. 그 부분의 코드를 File.Exists로 하지 말고 process.WaitForExit()로 바꿔서 처리하시면 좀더 안정적으로 동작할 것입니다.
정성태
2015-10-14 12시56분
[링크의 전설]

어둠이 걷히고 빛이 보이네요!!

AxImp.exe를 리플렉터로 까보면되는군요(아...생각해보면 당연한건데 답변을 보기전까진 미처 생각지 못했습니다;;하하).

아무튼 이제 어떻게든 해볼 수 있을 것 같습니다(일단은 시간 관계상 AxImp.exe를 사용할 것 같습니다만).

답변해주신 두분 정말 감사합니다.
[손님]

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5493xingTester5/6/2021160xing api XQCSPAT00600 질문입니다 [4]파일 다운로드1
5492한예지5/5/2021124FromCurrentSynchronizationContext 관련 코드 질문있습니다! [2]
5491조우성5/4/2021157WinForm과 WPF의 성능차이 비교한 자료가 있을까요? [3]
5490한예지5/3/2021119UI 스레드의 Invoke 질문있습니다. [4]
5489저누4/28/2021244시작하세요 C# 9.0, 225페이지 구조체 관련 질문드립니다. [2]
5488종범4/27/2021262C# 비동기 함수 async, await 와 Task의 관계에 대해 질문 드립니다. [6]
5487진우4/27/2021144C# 엑셀 자동화 성능 향상 문의 [2]
5486지나가던사람4/26/2021171닷넷 구현 코드 관련 질문 [2]
5485이재원4/17/2021241교재 315페이지 내용 질문 [3]
5484Syong4/16/2021222윈폼 기반의 응용프로그램 dll 참조와 32,64bit 빌드 관련 문의 [3]
5483한예지4/15/2021211익명 형식과 var 관계 질문 있습니다. [2]
5482질문4/13/2021245WPF를 위한 MVVM toolkit 선택과 관련한 문의드립니다. [2]
5480한예지4/5/2021416GetHashCode 질문있습니다! [2]
5479한예지4/4/2021242Equals를 닷넷에서 어떻게 구현했는지 보고 싶을 떄는 어떻게 해야 될까요? [2]
5478갑자기C#3/23/2021486C# Winform에서 TextBox없이 입력 받을 수 있나요? [7]
5477달팽이3/18/2021312c# 도형을 그린후 편집하는 방법이 궁금합니다. [2]파일 다운로드1
5475dimohy3/2/2021432POH가 .NET 5에 추가된 것으로 알고 있습니다. POH가 유용한 경우는 어떤 경우가 있을까요? [1]
5474Syong2/26/2021395사용자 지정 컨트롤 생성시 Invalidate, Update, Refresh의 차이점 [2]
5473한예지2/25/2021401디자인 타임이 뭔지 궁금합니다!! [5]
5472dimohy2/24/2021362등록하지 않고 ocx를 사용하는 `키움`관련 포스팅을 따라가다 질문드립니다. [2]
5471남산2/21/2021436MarshalDirectiveException 에 대한 질문입니다. [1]
5470한예지2/7/2021545yield 질문있습니다!! [2]
5469유호성2/6/2021475Parallel + Task.Run 동시 실행 환경에서 간헐적으로 Task.Run()에서 null이 리턴됩니다. [3]파일 다운로드1
5467heroho2/4/2021472실행 환경에 따른 Thread.Sleep 딜레이 차이 질문 [8]
5466pr1/29/2021569c# winform load시 작업표시줄에 뜨지 않는 현상을 겪으신적이 있으신가요? [4]파일 다운로드1
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...